- update on initial checkin (code is now same as in RileyLinkAAPS on 10.June - most command working)
- it will take some time until code is actually integrated into AAPS, but at least now all files are available
This commit is contained in:
parent
e8f860be8b
commit
9bae53e53a
277 changed files with 6819 additions and 14135 deletions
|
@ -57,7 +57,7 @@ def generateGitBuild = { ->
|
|||
return stringBuilder.toString()
|
||||
}
|
||||
|
||||
tasks.matching {it instanceof Test}.all {
|
||||
tasks.matching { it instanceof Test }.all {
|
||||
testLogging.events = ["failed", "skipped", "started"]
|
||||
testLogging.exceptionFormat = "full"
|
||||
}
|
||||
|
@ -97,7 +97,7 @@ android {
|
|||
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
|
||||
}
|
||||
debug {
|
||||
testCoverageEnabled (project.hasProperty('coverage') ? true : false)
|
||||
testCoverageEnabled(project.hasProperty('coverage') ? true : false)
|
||||
}
|
||||
}
|
||||
productFlavors {
|
||||
|
@ -164,7 +164,7 @@ android {
|
|||
unitTests.returnDefaultValues = true
|
||||
unitTests.includeAndroidResources = true
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
allprojects {
|
||||
repositories {
|
||||
|
@ -207,6 +207,7 @@ dependencies {
|
|||
exclude group: "com.google.android", module: "android"
|
||||
}
|
||||
implementation "org.apache.commons:commons-lang3:3.6"
|
||||
implementation "commons-collections:commons-collections:3.2.1"
|
||||
implementation "org.slf4j:slf4j-api:1.7.12"
|
||||
implementation "com.jjoe64:graphview:4.0.1"
|
||||
implementation "com.joanzapata.iconify:android-iconify-fontawesome:2.1.1"
|
||||
|
@ -252,7 +253,7 @@ dependencies {
|
|||
}
|
||||
|
||||
task unzip(type: Copy) {
|
||||
def zipPath = configurations.libs.find {it.name.startsWith("danars") }
|
||||
def zipPath = configurations.libs.find { it.name.startsWith("danars") }
|
||||
def zipFile = file(zipPath)
|
||||
def outputDir = file("${buildDir}/unpacked/dist")
|
||||
|
||||
|
|
|
@ -1,109 +0,0 @@
|
|||
package com.gxwtech.roundtrip2;
|
||||
|
||||
import android.content.res.Configuration;
|
||||
import android.os.Bundle;
|
||||
import android.preference.PreferenceActivity;
|
||||
import android.support.annotation.LayoutRes;
|
||||
import android.support.annotation.Nullable;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AppCompatDelegate;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.view.MenuInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
|
||||
/**
|
||||
* A {@link android.preference.PreferenceActivity} which implements and proxies the necessary calls
|
||||
* to be used with AppCompat.
|
||||
*/
|
||||
public abstract class AppCompatPreferenceActivity extends PreferenceActivity {
|
||||
|
||||
private AppCompatDelegate mDelegate;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
getDelegate().installViewFactory();
|
||||
getDelegate().onCreate(savedInstanceState);
|
||||
super.onCreate(savedInstanceState);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostCreate(Bundle savedInstanceState) {
|
||||
super.onPostCreate(savedInstanceState);
|
||||
getDelegate().onPostCreate(savedInstanceState);
|
||||
}
|
||||
|
||||
public ActionBar getSupportActionBar() {
|
||||
return getDelegate().getSupportActionBar();
|
||||
}
|
||||
|
||||
public void setSupportActionBar(@Nullable Toolbar toolbar) {
|
||||
getDelegate().setSupportActionBar(toolbar);
|
||||
}
|
||||
|
||||
@Override
|
||||
public MenuInflater getMenuInflater() {
|
||||
return getDelegate().getMenuInflater();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setContentView(@LayoutRes int layoutResID) {
|
||||
getDelegate().setContentView(layoutResID);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setContentView(View view) {
|
||||
getDelegate().setContentView(view);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setContentView(View view, ViewGroup.LayoutParams params) {
|
||||
getDelegate().setContentView(view, params);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void addContentView(View view, ViewGroup.LayoutParams params) {
|
||||
getDelegate().addContentView(view, params);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostResume() {
|
||||
super.onPostResume();
|
||||
getDelegate().onPostResume();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onTitleChanged(CharSequence title, int color) {
|
||||
super.onTitleChanged(title, color);
|
||||
getDelegate().setTitle(title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onConfigurationChanged(Configuration newConfig) {
|
||||
super.onConfigurationChanged(newConfig);
|
||||
getDelegate().onConfigurationChanged(newConfig);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onStop() {
|
||||
super.onStop();
|
||||
getDelegate().onStop();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
getDelegate().onDestroy();
|
||||
}
|
||||
|
||||
public void invalidateOptionsMenu() {
|
||||
getDelegate().invalidateOptionsMenu();
|
||||
}
|
||||
|
||||
private AppCompatDelegate getDelegate() {
|
||||
if (mDelegate == null) {
|
||||
mDelegate = AppCompatDelegate.create(this, null);
|
||||
}
|
||||
return mDelegate;
|
||||
}
|
||||
}
|
|
@ -1,21 +0,0 @@
|
|||
package com.gxwtech.roundtrip2;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
|
||||
import com.gxwtech.roundtrip2.CommunicationService.CommunicationService;
|
||||
|
||||
/**
|
||||
* Created by Tim on 07/06/2016.
|
||||
* Receives BOOT_COMPLETED Intent and starts service
|
||||
*/
|
||||
public class AutoStart extends BroadcastReceiver {
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
|
||||
context.startService(new Intent(context, CommunicationService.class));
|
||||
|
||||
}
|
||||
}
|
|
@ -1,279 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.CommunicationService;
|
||||
|
||||
import android.content.ComponentName;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.ServiceConnection;
|
||||
import android.content.res.Resources;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.IBinder;
|
||||
import android.os.Message;
|
||||
import android.os.Messenger;
|
||||
import android.os.RemoteException;
|
||||
import android.util.Log;
|
||||
import android.widget.Toast;
|
||||
|
||||
import com.google.gson.Gson;
|
||||
import com.google.gson.GsonBuilder;
|
||||
import com.google.gson.reflect.TypeToken;
|
||||
import com.gxwtech.roundtrip2.CommunicationService.Objects.Bolus;
|
||||
import com.gxwtech.roundtrip2.CommunicationService.Objects.DateDeserializer;
|
||||
import com.gxwtech.roundtrip2.CommunicationService.Objects.Integration;
|
||||
import com.gxwtech.roundtrip2.CommunicationService.Objects.IntegrationSerializer;
|
||||
import com.gxwtech.roundtrip2.CommunicationService.Objects.RealmManager;
|
||||
import com.gxwtech.roundtrip2.CommunicationService.Objects.TempBasal;
|
||||
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.util.Check;
|
||||
|
||||
|
||||
import java.util.Date;
|
||||
import java.util.List;
|
||||
|
||||
import info.nightscout.androidaps.MainApp;
|
||||
|
||||
/**
|
||||
* Created by Tim on 07/06/2016.
|
||||
* This service listens out for requests from HAPP and processes them
|
||||
*/
|
||||
public class CommunicationService extends android.app.Service {
|
||||
|
||||
public CommunicationService(){}
|
||||
final static String TAG = "CommunicationService";
|
||||
|
||||
class IncomingHandler extends Handler {
|
||||
@Override
|
||||
public void handleMessage(Message msg) {
|
||||
String action = "";
|
||||
Long requested = 0L;
|
||||
String safteyCheck = "";
|
||||
String pump = "";
|
||||
List<Integration> remoteIntegrations;
|
||||
List<String> remoteTreatments;
|
||||
Bundle data = new Bundle();
|
||||
RealmManager realmManager = new RealmManager();
|
||||
|
||||
Log.d(TAG, "START");
|
||||
try {
|
||||
|
||||
/*
|
||||
Expected Bundle data...
|
||||
"ACTION" - What is this incoming request? Example: "NEW_TREATMENTS"
|
||||
"DATE_REQUESTED" - When was this requested? So we can ignore old requests
|
||||
"PUMP" - Name of the pump the APS expects this app to support
|
||||
"INTEGRATION_OBJECTS" - Array of Integration Objects, details of the objects being synced. *OPTIONAL for NEW_TREATMENTS only*
|
||||
"TREATMENT_OBJECTS" - Array of Objects themselves being synced, TempBasal or Bolus *OPTIONAL for NEW_TREATMENTS only*
|
||||
*/
|
||||
|
||||
data = msg.getData();
|
||||
action = data.getString(RT2Const.commService.ACTION);
|
||||
requested = data.getLong(RT2Const.commService.DATE_REQUESTED, 0);
|
||||
pump = data.getString(RT2Const.commService.PUMP);
|
||||
Log.d("RECEIVED: ACTION", action);
|
||||
Log.d("RECEIVED: DATE", requested.toString());
|
||||
Log.d("RECEIVED: PUMP", pump);
|
||||
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
// TODO: 16/01/2016 Issue getting treatment details from APS app msg for user
|
||||
}
|
||||
|
||||
|
||||
switch (action) {
|
||||
case RT2Const.commService.INCOMING_TEST_MSG:
|
||||
Resources appR = MainApp.instance().getResources();
|
||||
CharSequence txt = appR.getText(appR.getIdentifier("app_name", "string", MainApp.instance().getPackageName()));
|
||||
Toast.makeText(MainApp.instance(), txt + ": Pump Driver App has connected successfully. ", Toast.LENGTH_LONG).show();
|
||||
Log.d(TAG, txt + ": APS app has connected successfully.");
|
||||
|
||||
break;
|
||||
case RT2Const.commService.INCOMING_NEW_TREATMENTS:
|
||||
GsonBuilder gsonBuilder = new GsonBuilder();
|
||||
gsonBuilder.registerTypeAdapter(Date.class, new DateDeserializer());
|
||||
Gson gson = gsonBuilder.create();
|
||||
|
||||
remoteIntegrations = gson.fromJson(data.getString(RT2Const.commService.INTEGRATION_OBJECTS), new TypeToken<List<Integration>>() {}.getType());
|
||||
remoteTreatments = gson.fromJson(data.getString(RT2Const.commService.TREATMENT_OBJECTS), new TypeToken<List<String>>() {}.getType());
|
||||
Log.d("RECEIVED: INTEGRATIONS", remoteIntegrations.toString());
|
||||
Log.d("RECEIVED: TREATMENTS", remoteTreatments.toString());
|
||||
|
||||
for (int i = 0; i < remoteIntegrations.size(); i++) {
|
||||
|
||||
realmManager.getRealm().beginTransaction();
|
||||
|
||||
Integration integrationForAPS = remoteIntegrations.get(i);
|
||||
integrationForAPS.setType ("aps_app");
|
||||
integrationForAPS.setState ("received");
|
||||
integrationForAPS.setToSync (true);
|
||||
integrationForAPS.setDate_updated (new Date());
|
||||
integrationForAPS.setRemote_id(remoteIntegrations.get(i).getLocal_object_id());
|
||||
|
||||
Integration integrationForPump = new Integration();
|
||||
integrationForPump.setType ("pump");
|
||||
integrationForPump.setDate_updated (new Date());
|
||||
integrationForPump.setLocal_object(remoteIntegrations.get(i).getLocal_object());
|
||||
|
||||
String localObjectlID = "", localObjectState = "", localObjectDetails = "", rejectRequest = "";
|
||||
if (!Check.isPumpSupported(pump))
|
||||
rejectRequest += "Pump requested not supported. ";
|
||||
if (Check.isRequestTooOld(requested)) rejectRequest += "Request too old. ";
|
||||
|
||||
switch (remoteIntegrations.get(i).getLocal_object()) {
|
||||
case "temp_basal":
|
||||
TempBasal tempBasal = gson.fromJson(remoteTreatments.get(i), TempBasal.class);
|
||||
realmManager.getRealm().copyToRealm(tempBasal);
|
||||
localObjectlID = tempBasal.getId();
|
||||
|
||||
switch (remoteIntegrations.get(i).getAction()) {
|
||||
case "new":
|
||||
rejectRequest += Check.isNewTempBasalSafe(tempBasal);
|
||||
if (rejectRequest.equals("")) // TODO: 12/08/2016 command to send TempBasal to pump
|
||||
break;
|
||||
case "cancel":
|
||||
rejectRequest += Check.isCancelTempBasalSafe(tempBasal, integrationForAPS, realmManager.getRealm());
|
||||
if (rejectRequest.equals("")) // TODO: 12/08/2016 command to send TempBasal to pump
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case "bolus_delivery":
|
||||
Bolus bolus = gson.fromJson(remoteTreatments.get(i), Bolus.class);
|
||||
realmManager.getRealm().copyToRealm(bolus);
|
||||
localObjectlID = bolus.getId();
|
||||
rejectRequest += Check.isBolusSafeToAction(bolus);
|
||||
|
||||
if (rejectRequest.equals("")) //TODO: 12/08/2016 command to action Bolus
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
if (rejectRequest.equals("")) {
|
||||
//all ok
|
||||
localObjectState = "received";
|
||||
localObjectDetails = "Request sent to pump";
|
||||
} else {
|
||||
//reject
|
||||
localObjectState = "error";
|
||||
localObjectDetails = rejectRequest;
|
||||
}
|
||||
|
||||
integrationForAPS.setLocal_object_id(localObjectlID);
|
||||
integrationForAPS.setState(localObjectState);
|
||||
integrationForAPS.setDetails(localObjectDetails);
|
||||
realmManager.getRealm().copyToRealm(integrationForAPS);
|
||||
integrationForPump.setLocal_object_id(localObjectlID);
|
||||
integrationForPump.setState(localObjectState);
|
||||
integrationForPump.setDetails(localObjectDetails);
|
||||
realmManager.getRealm().copyToRealm(integrationForPump);
|
||||
realmManager.getRealm().commitTransaction();
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
Log.e(TAG, "handleMessage: Unknown Action: " + action);
|
||||
}
|
||||
|
||||
connect_to_aps_app();
|
||||
realmManager.closeRealm();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
final Messenger myMessenger = new Messenger(new IncomingHandler());
|
||||
|
||||
@Override
|
||||
public IBinder onBind(Intent intent) {
|
||||
return myMessenger.getBinder();
|
||||
}
|
||||
|
||||
|
||||
public void updateAPSApp(){
|
||||
|
||||
RealmManager realmManager = new RealmManager();
|
||||
List<Integration> integrations = Integration.getIntegrationsToSync("aps_app", null, realmManager.getRealm());
|
||||
|
||||
if (integrations.size() > 0) {
|
||||
/*
|
||||
Bundle data...
|
||||
"ACTION" - What is this incoming request? Example: "TREATMENT_UPDATES"
|
||||
"INTEGRATION_OBJECTS" - Array of Integration Objects, details of the objects being synced. *OPTIONAL for UPDATE_TREATMENTS only*
|
||||
*/
|
||||
|
||||
Log.d(TAG, "UPDATE APS App:" + integrations.size() + " treatments to update");
|
||||
Log.d(TAG, "INTEGRATIONS:" + integrations.toString());
|
||||
Message msg = Message.obtain();
|
||||
boolean updateOK = true;
|
||||
try {
|
||||
Gson gson = new GsonBuilder()
|
||||
.registerTypeAdapter(Class.forName("io.realm.IntegrationRealmProxy"), new IntegrationSerializer())
|
||||
.create();
|
||||
|
||||
Bundle bundle = new Bundle();
|
||||
bundle.putString(RT2Const.commService.ACTION, RT2Const.commService.OUTGOING_TREATMENT_UPDATES);
|
||||
bundle.putString(RT2Const.commService.INTEGRATION_OBJECTS, gson.toJson(integrations));
|
||||
msg.setData(bundle);
|
||||
|
||||
} catch (ClassNotFoundException e){
|
||||
updateOK = false;
|
||||
Log.e(TAG, "Error creating gson object: " + e.getLocalizedMessage());
|
||||
}
|
||||
|
||||
try {
|
||||
myService.send(msg);
|
||||
Log.d(TAG, integrations.size() + " updates sent");
|
||||
} catch (RemoteException e) {
|
||||
updateOK = false;
|
||||
Log.e(TAG, integrations.size() + " updates failed. " + e.getLocalizedMessage());
|
||||
}
|
||||
|
||||
for (Integration integration : integrations){
|
||||
realmManager.getRealm().beginTransaction();
|
||||
if (updateOK) {
|
||||
integration.setState("sent");
|
||||
} else {
|
||||
integration.setState("error");
|
||||
integration.setDetails("Update to APS failed. Will not be resent.");
|
||||
}
|
||||
integration.setToSync(false);
|
||||
realmManager.getRealm().commitTransaction();
|
||||
}
|
||||
}
|
||||
|
||||
try {
|
||||
if (isBound) CommunicationService.this.unbindService(myConnection);
|
||||
} catch (IllegalArgumentException e) {
|
||||
//catch if service was killed in a unclean way
|
||||
}
|
||||
|
||||
realmManager.closeRealm();
|
||||
}
|
||||
|
||||
//Connect to the APS App Treatments Service
|
||||
private void connect_to_aps_app(){
|
||||
// TODO: 16/06/2016 should be able to pick the APS app from UI not hardcoded
|
||||
Intent intent = new Intent("com.hypodiabetic.happ.services.TreatmentService");
|
||||
intent.setPackage("com.hypodiabetic.happ");
|
||||
CommunicationService.this.bindService(intent, myConnection, Context.BIND_AUTO_CREATE);
|
||||
}
|
||||
//Our Service that APS App will connect to
|
||||
private Messenger myService = null;
|
||||
private Boolean isBound = false;
|
||||
private ServiceConnection myConnection = new ServiceConnection() {
|
||||
public void onServiceConnected(ComponentName className, IBinder service) {
|
||||
myService = new Messenger(service);
|
||||
isBound = true;
|
||||
|
||||
updateAPSApp();
|
||||
}
|
||||
|
||||
public void onServiceDisconnected(ComponentName className) {
|
||||
myService = null;
|
||||
isBound = false;
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
}
|
||||
|
|
@ -1,91 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.CommunicationService.Objects;
|
||||
|
||||
import java.util.Comparator;
|
||||
import java.util.Date;
|
||||
import java.util.List;
|
||||
import java.util.UUID;
|
||||
|
||||
import io.realm.Realm;
|
||||
import io.realm.RealmObject;
|
||||
import io.realm.RealmResults;
|
||||
import io.realm.Sort;
|
||||
|
||||
|
||||
/**
|
||||
* Created by Tim on 05/08/2016.
|
||||
*/
|
||||
public class Bolus extends RealmObject {
|
||||
|
||||
public Double getValue() {
|
||||
return value;
|
||||
}
|
||||
public void setValue(Double value) {
|
||||
this.value = value;
|
||||
}
|
||||
public void setTimestamp(Date timestamp) {
|
||||
this.timestamp = timestamp;
|
||||
}
|
||||
public Date getTimestamp() {
|
||||
return timestamp;
|
||||
}
|
||||
public String getType() {
|
||||
return type;
|
||||
}
|
||||
public void setType(String type) {
|
||||
this.type = type;
|
||||
}
|
||||
public String getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
private String id = UUID.randomUUID().toString();
|
||||
private Date timestamp = new Date();
|
||||
private String type;
|
||||
private Double value;
|
||||
|
||||
public static Bolus getBolus(String uuid, Realm realm) {
|
||||
RealmResults<Bolus> results = realm.where(Bolus.class)
|
||||
.equalTo("id", uuid)
|
||||
.findAllSorted("timestamp", Sort.DESCENDING);
|
||||
|
||||
if (results.isEmpty()) {
|
||||
return null;
|
||||
} else {
|
||||
return results.first();
|
||||
}
|
||||
}
|
||||
|
||||
public static List<Bolus> getBolusList(Realm realm){
|
||||
RealmResults<Bolus> results = realm.where(Bolus.class)
|
||||
.findAllSorted("timestamp", Sort.DESCENDING);
|
||||
if (results.isEmpty()) {
|
||||
return null;
|
||||
} else {
|
||||
return results;
|
||||
}
|
||||
}
|
||||
|
||||
public static List<Bolus> getBolusesBetween(Date dateFrom, Date dateTo, Realm realm) {
|
||||
RealmResults<Bolus> results = realm.where(Bolus.class)
|
||||
.greaterThanOrEqualTo("timestamp", dateFrom)
|
||||
.lessThanOrEqualTo("timestamp", dateTo)
|
||||
.findAllSorted("timestamp", Sort.DESCENDING);
|
||||
return results;
|
||||
}
|
||||
|
||||
public static Double getBolusCountBetween(Date dateFrom, Date dateTo, Realm realm) {
|
||||
Number result = realm.where(Bolus.class)
|
||||
.greaterThanOrEqualTo("timestamp", dateFrom)
|
||||
.lessThanOrEqualTo("timestamp", dateTo)
|
||||
.sum("value");
|
||||
return result.doubleValue();
|
||||
}
|
||||
|
||||
|
||||
public static class sortByDateTimeOld2YoungOLD implements Comparator<Bolus> {
|
||||
@Override
|
||||
public int compare(Bolus o1, Bolus o2) {
|
||||
return o2.getTimestamp().compareTo(o1.getTimestamp());
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,20 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.CommunicationService.Objects;
|
||||
|
||||
import com.google.gson.JsonDeserializationContext;
|
||||
import com.google.gson.JsonDeserializer;
|
||||
import com.google.gson.JsonElement;
|
||||
import java.lang.reflect.Type;
|
||||
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* Created by Tim on 16/08/2016.
|
||||
* Used by GSON to Deserializer Dates
|
||||
*/
|
||||
public class DateDeserializer implements JsonDeserializer<Date> {
|
||||
public Date deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) {
|
||||
Date date = null;
|
||||
date = new Date(json.getAsJsonPrimitive().getAsLong());
|
||||
return date;
|
||||
}
|
||||
}
|
|
@ -1,216 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.CommunicationService.Objects;
|
||||
|
||||
import java.util.Date;
|
||||
import java.util.List;
|
||||
import java.util.UUID;
|
||||
|
||||
import io.realm.Realm;
|
||||
import io.realm.RealmObject;
|
||||
import io.realm.RealmResults;
|
||||
import io.realm.Sort;
|
||||
|
||||
|
||||
/**
|
||||
* Created by Tim on 16/01/2016.
|
||||
* This table holds Integration details of an object
|
||||
* one object may have multiple Integrations
|
||||
*/
|
||||
|
||||
public class Integration extends RealmObject {
|
||||
|
||||
public String getId() {
|
||||
return id;
|
||||
}
|
||||
public String getType() {
|
||||
return type;
|
||||
}
|
||||
public void setType(String type) {
|
||||
this.type = type;
|
||||
}
|
||||
public String getState() {
|
||||
return state;
|
||||
}
|
||||
public void setState(String state) {
|
||||
this.state = state;
|
||||
}
|
||||
public String getAction() {
|
||||
return action;
|
||||
}
|
||||
public void setAction(String action) {
|
||||
this.action = action;
|
||||
}
|
||||
public Date getTimestamp() {
|
||||
return timestamp;
|
||||
}
|
||||
public Date getDate_updated() {
|
||||
return date_updated;
|
||||
}
|
||||
public void setDate_updated(Date date_updated) {
|
||||
this.date_updated = date_updated;
|
||||
}
|
||||
public String getLocal_object() {
|
||||
return local_object;
|
||||
}
|
||||
public void setLocal_object(String local_object) {
|
||||
this.local_object = local_object;
|
||||
}
|
||||
public String getLocal_object_id() {
|
||||
return local_object_id;
|
||||
}
|
||||
public void setLocal_object_id(String local_object_id) {
|
||||
this.local_object_id = local_object_id;
|
||||
}
|
||||
public String getRemote_id() {
|
||||
return remote_id;
|
||||
}
|
||||
public void setRemote_id(String remote_id) {
|
||||
this.remote_id = remote_id;
|
||||
}
|
||||
public String getDetails() {
|
||||
return details;
|
||||
}
|
||||
public void setDetails(String details) {
|
||||
this.details = details;
|
||||
}
|
||||
public String getRemote_var1() {
|
||||
return remote_var1;
|
||||
}
|
||||
public void setRemote_var1(String remote_var1) {
|
||||
this.remote_var1 = remote_var1;
|
||||
}
|
||||
public String getAuth_code() {
|
||||
return auth_code;
|
||||
}
|
||||
public void setAuth_code(String auth_code) {
|
||||
this.auth_code = auth_code;
|
||||
}
|
||||
public Boolean getToSync() {
|
||||
return toSync;
|
||||
}
|
||||
public void setToSync(Boolean toSync) {
|
||||
this.toSync = toSync;
|
||||
}
|
||||
|
||||
|
||||
private String id;
|
||||
private String type; //What Integration is this?
|
||||
private String state; //Current state this Integration is in
|
||||
private String action; //Requested action for this object
|
||||
private Date timestamp; //Date created
|
||||
private Date date_updated; //Last time the Integration for this object was updated
|
||||
private String local_object; //What rt2 object is this? Bolus, TempBasal etc
|
||||
private String local_object_id; //HAPP ID for this object
|
||||
private String remote_id; //ID provided by the remote system
|
||||
private String details; //The details of this Integration attempt
|
||||
private String remote_var1; //Misc information about this Integration
|
||||
private String auth_code; //auth_code if required
|
||||
private Boolean toSync; //Do we need to sync this object?
|
||||
|
||||
public Integration(){
|
||||
id = UUID.randomUUID().toString();
|
||||
timestamp = new Date();
|
||||
date_updated = new Date();
|
||||
remote_var1 = "";
|
||||
state = "";
|
||||
toSync = true;
|
||||
}
|
||||
|
||||
public Integration(String type, String local_object, String local_object_id){
|
||||
id = UUID.randomUUID().toString();
|
||||
timestamp = new Date();
|
||||
date_updated = new Date();
|
||||
remote_var1 = "";
|
||||
state = "";
|
||||
this.type = type;
|
||||
this.local_object = local_object;
|
||||
this.local_object_id = local_object_id;
|
||||
toSync = true;
|
||||
}
|
||||
|
||||
public static Integration getIntegration(String type, String local_object, String rt2_id, Realm realm){
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("type", type)
|
||||
.equalTo("local_object", local_object)
|
||||
.equalTo("local_object_id", rt2_id)
|
||||
.findAllSorted("date_updated", Sort.DESCENDING);
|
||||
|
||||
if (results.isEmpty()) { //We dont have an Integration for this item
|
||||
return null;
|
||||
|
||||
} else { //Found an Integration, return it
|
||||
return results.first();
|
||||
}
|
||||
}
|
||||
|
||||
public static List<Integration> getIntegrationsFor(String local_object, String local_object_id, Realm realm) {
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("local_object", local_object)
|
||||
.equalTo("local_object_id", local_object_id)
|
||||
.findAllSorted("date_updated", Sort.DESCENDING);
|
||||
return results;
|
||||
}
|
||||
|
||||
public static List<Integration> getIntegrationsHoursOld(String type, String local_object, int inLastHours, Realm realm) {
|
||||
Date now = new Date();
|
||||
Date hoursAgo = new Date(now.getTime() - (inLastHours * 60 * 60 * 1000));
|
||||
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("local_object", local_object)
|
||||
.equalTo("type", type)
|
||||
.greaterThanOrEqualTo("date_updated", hoursAgo)
|
||||
.lessThanOrEqualTo("date_updated", now)
|
||||
.findAllSorted("date_updated", Sort.DESCENDING);
|
||||
return results;
|
||||
}
|
||||
|
||||
public static List<Integration> getIntegrationsToSync(String type, String local_object, Realm realm) {
|
||||
if (local_object != null) {
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("local_object", local_object)
|
||||
.equalTo("type", type)
|
||||
.equalTo("toSync", Boolean.TRUE)
|
||||
.findAllSorted("date_updated", Sort.DESCENDING);
|
||||
return results;
|
||||
} else {
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("type", type)
|
||||
.equalTo("toSync", Boolean.TRUE)
|
||||
.findAllSorted("date_updated", Sort.DESCENDING);
|
||||
return results;
|
||||
}
|
||||
}
|
||||
|
||||
public static Integration getIntegrationByID(String uuid, Realm realm) {
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("id", uuid)
|
||||
.findAllSorted("timestamp", Sort.DESCENDING);
|
||||
if (results.isEmpty()) {
|
||||
return null;
|
||||
} else {
|
||||
return results.first();
|
||||
}
|
||||
}
|
||||
|
||||
public static List<Integration> getUpdatedInLastMins(Integer inLastMins, String type, Realm realm) {
|
||||
Date now = new Date();
|
||||
Date minsAgo = new Date(now.getTime() - (inLastMins * 60 * 1000));
|
||||
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("type", type)
|
||||
.greaterThanOrEqualTo("date_updated", minsAgo)
|
||||
.lessThanOrEqualTo("date_updated", now)
|
||||
.findAllSorted("date_updated", Sort.DESCENDING);
|
||||
return results;
|
||||
}
|
||||
|
||||
public static List<Integration> getIntegrationsWithErrors(String type, Realm realm) {
|
||||
RealmResults<Integration> results = realm.where(Integration.class)
|
||||
.equalTo("type", type)
|
||||
.equalTo("state", "error")
|
||||
.findAllSorted("date_updated", Sort.DESCENDING);
|
||||
return results;
|
||||
}
|
||||
|
||||
|
||||
|
||||
}
|
|
@ -1,35 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.CommunicationService.Objects;
|
||||
|
||||
import com.google.gson.JsonElement;
|
||||
import com.google.gson.JsonObject;
|
||||
import com.google.gson.JsonSerializationContext;
|
||||
import com.google.gson.JsonSerializer;
|
||||
|
||||
import java.lang.reflect.Type;
|
||||
|
||||
/**
|
||||
* Created by Tim on 12/08/2016.
|
||||
* Required by Realm for converting to gson https://realm.io/docs/java/latest/#gson
|
||||
*/
|
||||
public class IntegrationSerializer implements JsonSerializer<Integration> {
|
||||
|
||||
@Override
|
||||
public JsonElement serialize(Integration src, Type typeOfSrc, JsonSerializationContext context) {
|
||||
final JsonObject jsonObject = new JsonObject();
|
||||
jsonObject.addProperty("id", src.getId());
|
||||
jsonObject.addProperty("type", src.getType());
|
||||
jsonObject.addProperty("state", src.getState());
|
||||
jsonObject.addProperty("action", src.getAction());
|
||||
jsonObject.addProperty("timestamp", src.getTimestamp().getTime());
|
||||
jsonObject.addProperty("date_updated", src.getDate_updated().getTime());
|
||||
jsonObject.addProperty("local_object", src.getLocal_object());
|
||||
jsonObject.addProperty("local_object_id", src.getLocal_object_id());
|
||||
jsonObject.addProperty("remote_id", src.getRemote_id());
|
||||
jsonObject.addProperty("details", src.getDetails());
|
||||
jsonObject.addProperty("remote_var1", src.getRemote_var1());
|
||||
jsonObject.addProperty("auth_code", src.getAuth_code());
|
||||
jsonObject.addProperty("toSync", src.getToSync());
|
||||
|
||||
return jsonObject;
|
||||
}
|
||||
}
|
|
@ -1,23 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.CommunicationService.Objects;
|
||||
|
||||
import io.realm.Realm;
|
||||
|
||||
/**
|
||||
* Created by Tim on 11/08/2016.
|
||||
*/
|
||||
public class RealmManager {
|
||||
private Realm realm;
|
||||
|
||||
public RealmManager(){
|
||||
realm = Realm.getDefaultInstance();
|
||||
}
|
||||
|
||||
public void closeRealm(){
|
||||
realm.close();
|
||||
}
|
||||
|
||||
public Realm getRealm(){
|
||||
if (realm.isClosed() || realm.isEmpty()) realm = Realm.getDefaultInstance();
|
||||
return realm;
|
||||
}
|
||||
}
|
|
@ -1,179 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.CommunicationService.Objects;
|
||||
|
||||
import java.util.Date;
|
||||
import java.util.List;
|
||||
import java.util.UUID;
|
||||
|
||||
import io.realm.Realm;
|
||||
import io.realm.RealmObject;
|
||||
import io.realm.RealmResults;
|
||||
import io.realm.Sort;
|
||||
import io.realm.annotations.Ignore;
|
||||
|
||||
/**
|
||||
* Created by Tim on 03/09/2015.
|
||||
*/
|
||||
public class TempBasal extends RealmObject {
|
||||
|
||||
|
||||
public Double getRate() {
|
||||
return rate;
|
||||
}
|
||||
public void setRate(Double rate) {
|
||||
this.rate = rate;
|
||||
}
|
||||
public Integer getDuration() {
|
||||
return duration;
|
||||
}
|
||||
public void setDuration(Integer duration) {
|
||||
this.duration = duration;
|
||||
}
|
||||
public Date getStart_time() {
|
||||
return start_time;
|
||||
}
|
||||
public void setStart_time(Date start_time) {
|
||||
this.start_time = start_time;
|
||||
}
|
||||
public String getBasal_adjustemnt() {
|
||||
return basal_adjustemnt;
|
||||
}
|
||||
public void setBasal_adjustemnt(String basal_adjustemnt) {
|
||||
this.basal_adjustemnt = basal_adjustemnt;
|
||||
}
|
||||
public String getAps_mode() {
|
||||
return aps_mode;
|
||||
}
|
||||
public void setAps_mode(String aps_mode) {
|
||||
this.aps_mode = aps_mode;
|
||||
}
|
||||
public String getId() {
|
||||
return id;
|
||||
}
|
||||
public Date getTimestamp() {
|
||||
return timestamp;
|
||||
}
|
||||
|
||||
private String id = UUID.randomUUID().toString();
|
||||
private Double rate = 0D; //Temp Basal Rate for (U/hr) mode
|
||||
private Integer duration = 0; //Duration of Temp
|
||||
private Date start_time; //When the Temp Basal started
|
||||
private String basal_adjustemnt = ""; //High or Low temp
|
||||
private String aps_mode;
|
||||
|
||||
@Ignore
|
||||
public Date timestamp = new Date();
|
||||
|
||||
public static TempBasal getTempBasalByID(String uuid, Realm realm) {
|
||||
RealmResults<TempBasal> results = realm.where(TempBasal.class)
|
||||
.equalTo("id", uuid)
|
||||
.findAllSorted("start_time", Sort.DESCENDING);
|
||||
|
||||
if (results.isEmpty()) {
|
||||
return null;
|
||||
} else {
|
||||
return results.first();
|
||||
}
|
||||
}
|
||||
|
||||
public static TempBasal last(Realm realm) {
|
||||
RealmResults<TempBasal> results = realm.where(TempBasal.class)
|
||||
.findAllSorted("start_time", Sort.DESCENDING);
|
||||
|
||||
if (results.isEmpty()) {
|
||||
return new TempBasal(); //returns an empty TempBasal, other than null
|
||||
} else {
|
||||
return results.first();
|
||||
}
|
||||
}
|
||||
|
||||
public static TempBasal lastActive(Realm realm) {
|
||||
RealmResults<TempBasal> results = realm.where(TempBasal.class)
|
||||
.findAllSorted("start_time", Sort.DESCENDING);
|
||||
|
||||
if (results.isEmpty()) {
|
||||
return null;
|
||||
} else {
|
||||
TempBasal tempBasal = results.first();
|
||||
Integration integration = Integration.getIntegration("pump","temp_Basal",tempBasal.getId(),realm);
|
||||
|
||||
if (integration.getState().equals("set")){
|
||||
return tempBasal;
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static TempBasal getCurrentActive(Date atThisDate, Realm realm) {
|
||||
RealmResults<TempBasal> results = realm.where(TempBasal.class)
|
||||
.findAllSorted("start_time", Sort.DESCENDING);
|
||||
|
||||
TempBasal last = null;
|
||||
if (!results.isEmpty()) last = results.first();
|
||||
if (last != null && last.isactive(atThisDate)){
|
||||
return last;
|
||||
} else {
|
||||
return new TempBasal(); //returns an empty TempBasal, other than null or inactive basal
|
||||
}
|
||||
}
|
||||
|
||||
public boolean isactive(Date atThisDate){
|
||||
if (atThisDate == null) atThisDate = new Date();
|
||||
|
||||
if (start_time == null){ return false;}
|
||||
|
||||
Date fur = new Date(start_time.getTime() + duration * 60000);
|
||||
if (fur.after(atThisDate)){
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public String ageFormattted(){
|
||||
Integer minsOld = age();
|
||||
if (minsOld > 1){
|
||||
return minsOld + " mins ago";
|
||||
} else {
|
||||
return minsOld + " min ago";
|
||||
}
|
||||
}
|
||||
|
||||
public int age(){
|
||||
Date timeNow = new Date();
|
||||
return (int)(timeNow.getTime() - timestamp.getTime()) /1000/60; //Age in Mins the Temp Basal was suggested
|
||||
}
|
||||
|
||||
public Date endDate(){
|
||||
Date endedAt = new Date(start_time.getTime() + (duration * 1000 * 60)); //The date this Temp Basal ended
|
||||
return endedAt;
|
||||
}
|
||||
|
||||
public Long durationLeft(){
|
||||
if (start_time != null) {
|
||||
Date timeNow = new Date();
|
||||
Long min_left = ((start_time.getTime() + duration * 60000) - timeNow.getTime()) / 60000; //Time left to run in Mins
|
||||
return min_left;
|
||||
} else {
|
||||
return duration.longValue();
|
||||
}
|
||||
}
|
||||
|
||||
public static List<TempBasal> getTempBasalsDated(Date dateFrom, Date dateTo, Realm realm) {
|
||||
RealmResults<TempBasal> results = realm.where(TempBasal.class)
|
||||
.greaterThanOrEqualTo("start_time", dateFrom)
|
||||
.lessThanOrEqualTo("start_time", dateTo)
|
||||
.findAllSorted("start_time", Sort.DESCENDING);
|
||||
return results;
|
||||
}
|
||||
|
||||
public boolean checkIsCancelRequest() {
|
||||
if (rate.equals(0D) && duration.equals(0)){
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
@ -1,83 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.HistoryActivity;
|
||||
|
||||
import android.content.Intent;
|
||||
import android.os.Bundle;
|
||||
import android.support.design.widget.FloatingActionButton;
|
||||
import android.support.design.widget.Snackbar;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.view.MenuItem;
|
||||
import android.view.View;
|
||||
|
||||
import com.gxwtech.roundtrip2.R;
|
||||
|
||||
/**
|
||||
* An activity representing a single HistoryPage detail screen. This
|
||||
* activity is only used narrow width devices. On tablet-size devices,
|
||||
* item details are presented side-by-side with a list of items
|
||||
* in a {@link HistoryPageListActivity}.
|
||||
*/
|
||||
public class HistoryPageDetailActivity extends AppCompatActivity {
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_historypage_detail);
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.detail_toolbar);
|
||||
setSupportActionBar(toolbar);
|
||||
|
||||
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
|
||||
fab.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
Snackbar.make(view, "Replace with your own detail action", Snackbar.LENGTH_LONG)
|
||||
.setAction("Action", null).show();
|
||||
}
|
||||
});
|
||||
|
||||
// Show the Up button in the action bar.
|
||||
ActionBar actionBar = getSupportActionBar();
|
||||
if (actionBar != null) {
|
||||
actionBar.setDisplayHomeAsUpEnabled(true);
|
||||
}
|
||||
|
||||
// savedInstanceState is non-null when there is fragment state
|
||||
// saved from previous configurations of this activity
|
||||
// (e.g. when rotating the screen from portrait to landscape).
|
||||
// In this case, the fragment will automatically be re-added
|
||||
// to its container so we don't need to manually add it.
|
||||
// For more information, see the Fragments API guide at:
|
||||
//
|
||||
// http://developer.android.com/guide/components/fragments.html
|
||||
//
|
||||
if (savedInstanceState == null) {
|
||||
// Create the detail fragment and add it to the activity
|
||||
// using a fragment transaction.
|
||||
Bundle arguments = new Bundle();
|
||||
arguments.putString(HistoryPageDetailFragment.ARG_ITEM_ID,
|
||||
getIntent().getStringExtra(HistoryPageDetailFragment.ARG_ITEM_ID));
|
||||
HistoryPageDetailFragment fragment = new HistoryPageDetailFragment();
|
||||
fragment.setArguments(arguments);
|
||||
getSupportFragmentManager().beginTransaction()
|
||||
.add(R.id.historypage_detail_container, fragment)
|
||||
.commit();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onOptionsItemSelected(MenuItem item) {
|
||||
int id = item.getItemId();
|
||||
if (id == android.R.id.home) {
|
||||
// This ID represents the Home or Up button. In the case of this
|
||||
// activity, the Up button is shown. For
|
||||
// more details, see the Navigation pattern on Android Design:
|
||||
//
|
||||
// http://developer.android.com/design/patterns/navigation.html#up-vs-back
|
||||
//
|
||||
navigateUpTo(new Intent(this, HistoryPageListActivity.class));
|
||||
return true;
|
||||
}
|
||||
return super.onOptionsItemSelected(item);
|
||||
}
|
||||
}
|
|
@ -1,70 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.HistoryActivity;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.os.Bundle;
|
||||
import android.support.design.widget.CollapsingToolbarLayout;
|
||||
import android.support.v4.app.Fragment;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.TextView;
|
||||
|
||||
import com.gxwtech.roundtrip2.R;
|
||||
|
||||
|
||||
/**
|
||||
* A fragment representing a single HistoryPage detail screen.
|
||||
* This fragment is either contained in a {@link HistoryPageListActivity}
|
||||
* in two-pane mode (on tablets) or a {@link HistoryPageDetailActivity}
|
||||
* on handsets.
|
||||
*/
|
||||
public class HistoryPageDetailFragment extends Fragment {
|
||||
/**
|
||||
* The fragment argument representing the item ID that this fragment
|
||||
* represents.
|
||||
*/
|
||||
public static final String ARG_ITEM_ID = "item_id";
|
||||
|
||||
/**
|
||||
* The dummy content this fragment is presenting.
|
||||
*/
|
||||
private HistoryPageListContent.RecordHolder mItem;
|
||||
|
||||
/**
|
||||
* Mandatory empty constructor for the fragment manager to instantiate the
|
||||
* fragment (e.g. upon screen orientation changes).
|
||||
*/
|
||||
public HistoryPageDetailFragment() {
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
|
||||
if (getArguments().containsKey(ARG_ITEM_ID)) {
|
||||
// Load the dummy content specified by the fragment
|
||||
// arguments. In a real-world scenario, use a Loader
|
||||
// to load content from a content provider.
|
||||
mItem = HistoryPageListContent.ITEM_MAP.get(getArguments().getString(ARG_ITEM_ID));
|
||||
|
||||
Activity activity = this.getActivity();
|
||||
CollapsingToolbarLayout appBarLayout = (CollapsingToolbarLayout) activity.findViewById(R.id.toolbar_layout);
|
||||
if (appBarLayout != null) {
|
||||
appBarLayout.setTitle("Details"/*mItem.content*/);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
||||
Bundle savedInstanceState) {
|
||||
View rootView = inflater.inflate(R.layout.historypage_detail, container, false);
|
||||
|
||||
// Show the dummy content as text in a TextView.
|
||||
if (mItem != null) {
|
||||
((TextView) rootView.findViewById(R.id.historypage_detail)).setText(mItem.details);
|
||||
}
|
||||
|
||||
return rootView;
|
||||
}
|
||||
}
|
|
@ -1,230 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.HistoryActivity;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.os.Bundle;
|
||||
import android.support.annotation.NonNull;
|
||||
import android.support.design.widget.FloatingActionButton;
|
||||
import android.support.design.widget.Snackbar;
|
||||
import android.support.v4.content.LocalBroadcastManager;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.RecyclerView;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.util.Log;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.Menu;
|
||||
import android.view.MenuItem;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.TextView;
|
||||
|
||||
import com.gxwtech.roundtrip2.R;
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.ServiceData.RetrieveHistoryPageResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.Set;
|
||||
|
||||
/**
|
||||
* An activity representing a list of HistoryPages. This activity
|
||||
* has different presentations for handset and tablet-size devices. On
|
||||
* handsets, the activity presents a list of items, which when touched,
|
||||
* lead to a {@link HistoryPageDetailActivity} representing
|
||||
* item details. On tablets, the activity presents the list of items and
|
||||
* item details side-by-side using two vertical panes.
|
||||
*/
|
||||
public class HistoryPageListActivity extends AppCompatActivity {
|
||||
private static final String TAG = "HistoryPageListActivity";
|
||||
|
||||
|
||||
/**
|
||||
* Whether or not the activity is in two-pane mode, i.e. running on a tablet
|
||||
* device.
|
||||
*/
|
||||
private boolean mTwoPane;
|
||||
private BroadcastReceiver mBroadcastRecevier;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_historypage_list);
|
||||
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
setSupportActionBar(toolbar);
|
||||
toolbar.setTitle(R.string.title_pump_history);
|
||||
|
||||
// Show the Up button in the action bar.
|
||||
ActionBar actionBar = getSupportActionBar();
|
||||
if (actionBar != null) {
|
||||
actionBar.setDisplayHomeAsUpEnabled(true);
|
||||
}
|
||||
|
||||
View recyclerView = findViewById(R.id.historypage_list);
|
||||
assert recyclerView != null;
|
||||
setupRecyclerView((RecyclerView) recyclerView);
|
||||
|
||||
if (findViewById(R.id.historypage_detail_container) != null) {
|
||||
// The detail container view will be present only in the
|
||||
// large-screen layouts (res/values-w900dp).
|
||||
// If this view is present, then the
|
||||
// activity should be in two-pane mode.
|
||||
mTwoPane = true;
|
||||
}
|
||||
|
||||
mBroadcastRecevier = new BroadcastReceiver() {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent receivedIntent) {
|
||||
if (receivedIntent == null) {
|
||||
Log.e(TAG,"onReceive: received null intent");
|
||||
} else {
|
||||
String action = receivedIntent.getAction();
|
||||
if (action == null) {
|
||||
Log.e(TAG, "onReceive: null action");
|
||||
} else {
|
||||
if (RT2Const.local.INTENT_historyPageBundleIncoming.equals(action)) {
|
||||
Bundle incomingBundle = receivedIntent.getExtras().getBundle(RT2Const.IPC.MSG_PUMP_history_key);
|
||||
ServiceTransport transport = new ServiceTransport(incomingBundle);
|
||||
ServiceResult result = transport.getServiceResult();
|
||||
if ("RetrieveHistoryPageResult".equals(result.getServiceResultType())) {
|
||||
RetrieveHistoryPageResult pageResult = (RetrieveHistoryPageResult) result;
|
||||
Bundle page = pageResult.getPageBundle();
|
||||
ArrayList<Bundle> recordBundleList = page.getParcelableArrayList("mRecordList");
|
||||
try {
|
||||
for (Bundle record : recordBundleList) {
|
||||
HistoryPageListContent.addItem(record);
|
||||
}
|
||||
} catch (java.lang.NullPointerException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG,"Unrecognized intent action: "+action);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
IntentFilter filter = new IntentFilter();
|
||||
filter.addAction(RT2Const.local.INTENT_historyPageBundleIncoming);
|
||||
LocalBroadcastManager.getInstance(getApplicationContext()).registerReceiver(mBroadcastRecevier,filter);
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCreateOptionsMenu(Menu menu) {
|
||||
getMenuInflater().inflate(R.menu.menu_bluetooth_scan, menu);
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onOptionsItemSelected(MenuItem item) {
|
||||
switch (item.getItemId()) {
|
||||
case R.id.miScan:
|
||||
getHistory();
|
||||
return true;
|
||||
default:
|
||||
return super.onOptionsItemSelected(item);
|
||||
}
|
||||
}
|
||||
|
||||
public void getHistory(){
|
||||
// tell them we're ready for data
|
||||
Intent intent = new Intent(RT2Const.local.INTENT_historyPageViewerReady);
|
||||
LocalBroadcastManager.getInstance(getApplicationContext()).sendBroadcast(intent);
|
||||
}
|
||||
|
||||
private void setupRecyclerView(@NonNull RecyclerView recyclerView) {
|
||||
recyclerView.setAdapter(new SimpleItemRecyclerViewAdapter(HistoryPageListContent.ITEMS));
|
||||
}
|
||||
|
||||
public class SimpleItemRecyclerViewAdapter
|
||||
extends RecyclerView.Adapter<SimpleItemRecyclerViewAdapter.ViewHolder> {
|
||||
|
||||
private final List<HistoryPageListContent.RecordHolder> mValues;
|
||||
|
||||
public SimpleItemRecyclerViewAdapter(List<HistoryPageListContent.RecordHolder> items) {
|
||||
mValues = items;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
|
||||
View view = LayoutInflater.from(parent.getContext())
|
||||
.inflate(R.layout.historypage_list_content, parent, false);
|
||||
return new ViewHolder(view);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBindViewHolder(final ViewHolder holder, int position) {
|
||||
holder.mItem = mValues.get(position);
|
||||
holder.mIdView.setText(mValues.get(position).dateAndName);
|
||||
String keytext = "";
|
||||
Set<String> keys = holder.mItem.content.keySet();
|
||||
int n = 0;
|
||||
for (String key : keys) {
|
||||
if (!key.equals("_type") && !key.equals("_stype") && !key.equals("timestamp") && !key.equals("_opcode")) {
|
||||
try {
|
||||
keytext += key + ":" + holder.mItem.content.get(key).toString();
|
||||
n++;
|
||||
if (n < keys.size() - 1) {
|
||||
keytext += "\n";
|
||||
}
|
||||
} catch (NullPointerException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
holder.mContentView.setText(keytext);
|
||||
|
||||
holder.mView.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View v) {
|
||||
if (mTwoPane) {
|
||||
Bundle arguments = new Bundle();
|
||||
arguments.putString(HistoryPageDetailFragment.ARG_ITEM_ID, holder.mItem.id);
|
||||
HistoryPageDetailFragment fragment = new HistoryPageDetailFragment();
|
||||
fragment.setArguments(arguments);
|
||||
getSupportFragmentManager().beginTransaction()
|
||||
.replace(R.id.historypage_detail_container, fragment)
|
||||
.commit();
|
||||
} else {
|
||||
Context context = v.getContext();
|
||||
Intent intent = new Intent(context, HistoryPageDetailActivity.class);
|
||||
intent.putExtra(HistoryPageDetailFragment.ARG_ITEM_ID, holder.mItem.id);
|
||||
|
||||
context.startActivity(intent);
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getItemCount() {
|
||||
return mValues.size();
|
||||
}
|
||||
|
||||
public class ViewHolder extends RecyclerView.ViewHolder {
|
||||
public final View mView;
|
||||
public final TextView mIdView;
|
||||
public final TextView mContentView;
|
||||
public HistoryPageListContent.RecordHolder mItem;
|
||||
|
||||
public ViewHolder(View view) {
|
||||
super(view);
|
||||
mView = view;
|
||||
mIdView = (TextView) view.findViewById(R.id.id);
|
||||
mContentView = (TextView) view.findViewById(R.id.content);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return super.toString() + " '" + mContentView.getText() + "'";
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,94 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.HistoryActivity;
|
||||
|
||||
import android.os.Bundle;
|
||||
import android.util.ArraySet;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
import java.util.HashSet;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/12/16.
|
||||
*/
|
||||
public class HistoryPageListContent {
|
||||
public static final List<RecordHolder> ITEMS = new ArrayList<>();
|
||||
|
||||
/**
|
||||
* A map of items, by ID.
|
||||
*/
|
||||
public static final Map<String, RecordHolder> ITEM_MAP = new HashMap<>();
|
||||
|
||||
static void addItem(Bundle recordBundle) {
|
||||
addItem(new RecordHolder(recordBundle));
|
||||
}
|
||||
|
||||
private static void addItem(RecordHolder item) {
|
||||
ITEMS.add(item);
|
||||
ITEM_MAP.put(item.id, item);
|
||||
}
|
||||
|
||||
private static String makeDetails(int position) {
|
||||
RecordHolder rh = ITEMS.get(position);
|
||||
if (rh == null) {
|
||||
return "(null)";
|
||||
}
|
||||
|
||||
return makeDetails(rh.content);
|
||||
}
|
||||
|
||||
private static String makeDetails(Bundle historyEntry) {
|
||||
Set<String> ignoredSet = new HashSet<>();
|
||||
ignoredSet.add("_type");
|
||||
ignoredSet.add("_stype");
|
||||
ignoredSet.add("_opcode");
|
||||
ignoredSet.add("timestamp");
|
||||
StringBuilder builder = new StringBuilder();
|
||||
int n = 0;
|
||||
for (String key : historyEntry.keySet()) {
|
||||
if (!ignoredSet.contains(key)) {
|
||||
builder.append(key);
|
||||
n++;
|
||||
if (n<historyEntry.keySet().size()-1) {
|
||||
builder.append("\n");
|
||||
}
|
||||
}
|
||||
}
|
||||
return builder.toString();
|
||||
}
|
||||
|
||||
public static class RecordHolder {
|
||||
public final String id;
|
||||
public final String dateAndName;
|
||||
public final Bundle content;
|
||||
public final String details;
|
||||
|
||||
public RecordHolder(Bundle content) {
|
||||
id = String.format("%d",content.hashCode());
|
||||
String rawTimestamp = content.getString("timestamp","0000-00-00T00:00:00");
|
||||
int tspot = rawTimestamp.indexOf('T');
|
||||
StringBuilder dateAndNameBuilder = new StringBuilder();
|
||||
dateAndNameBuilder.append(rawTimestamp.substring(0,tspot-1));
|
||||
dateAndNameBuilder.append("\n");
|
||||
dateAndNameBuilder.append(rawTimestamp.substring(tspot+1,rawTimestamp.length()-1));
|
||||
dateAndNameBuilder.append("\n");
|
||||
String veryShortName = content.getString("_stype","(??????)");
|
||||
if (veryShortName.length() >= 12) {
|
||||
veryShortName = veryShortName.substring(0,12);
|
||||
}
|
||||
dateAndNameBuilder.append(veryShortName);
|
||||
this.dateAndName = dateAndNameBuilder.toString();
|
||||
this.content = content;
|
||||
details = makeDetails(content);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return content.getString("_stype", "(unk)");
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
@ -1,614 +0,0 @@
|
|||
package com.gxwtech.roundtrip2;
|
||||
|
||||
import android.bluetooth.BluetoothAdapter;
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.ComponentName;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.content.ServiceConnection;
|
||||
import android.content.res.Resources;
|
||||
import android.graphics.PorterDuff;
|
||||
import android.graphics.drawable.Drawable;
|
||||
import android.os.Bundle;
|
||||
import android.os.IBinder;
|
||||
import android.os.Message;
|
||||
import android.os.Messenger;
|
||||
import android.os.RemoteException;
|
||||
import android.os.SystemClock;
|
||||
import android.support.v4.content.LocalBroadcastManager;
|
||||
import android.support.v4.view.GravityCompat;
|
||||
import android.support.v4.widget.DrawerLayout;
|
||||
import android.support.v7.app.ActionBarDrawerToggle;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.design.widget.Snackbar;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.util.Log;
|
||||
import android.view.KeyEvent;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.Menu;
|
||||
import android.view.MenuItem;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.AdapterView;
|
||||
import android.widget.BaseAdapter;
|
||||
import android.widget.ImageView;
|
||||
import android.widget.LinearLayout;
|
||||
import android.widget.ListView;
|
||||
import android.widget.ProgressBar;
|
||||
import android.widget.TextView;
|
||||
import android.preference.PreferenceManager;
|
||||
import android.content.SharedPreferences;
|
||||
|
||||
import com.gxwtech.roundtrip2.HistoryActivity.HistoryPageListActivity;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.ListView;
|
||||
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.ServiceData.BasalProfile;
|
||||
import com.gxwtech.roundtrip2.ServiceData.BolusWizardCarbProfile;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ISFProfile;
|
||||
import com.gxwtech.roundtrip2.ServiceData.PumpModelResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ReadPumpClockResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceClientActions;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceCommand;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceNotification;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
import com.gxwtech.roundtrip2.ServiceMessageViewActivity.ServiceMessageViewListActivity;
|
||||
import com.gxwtech.roundtrip2.util.tools;
|
||||
|
||||
import java.util.ArrayList;
|
||||
|
||||
public class MainActivity extends AppCompatActivity {
|
||||
|
||||
private static final String TAG = "MainActivity";
|
||||
private static final int REQUEST_ENABLE_BT = 2177; // just something unique.
|
||||
private RoundtripServiceClientConnection roundtripServiceClientConnection;
|
||||
private BroadcastReceiver mBroadcastReceiver;
|
||||
|
||||
BroadcastReceiver apsAppConnected;
|
||||
Bundle storeForHistoryViewer;
|
||||
|
||||
//UI items
|
||||
private DrawerLayout mDrawerLayout;
|
||||
private LinearLayout mDrawerLinear;
|
||||
private Toolbar toolbar;
|
||||
|
||||
public static Context mContext; // TODO: 09/07/2016 @TIM this should not be needed
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
Log.w(TAG,"onCreate");
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_main);
|
||||
|
||||
setupMenuAndToolbar();
|
||||
|
||||
mContext = this; // TODO: 09/07/2016 @TIM this should not be needed
|
||||
|
||||
//Sets default Preferences
|
||||
PreferenceManager.setDefaultValues(this, R.xml.pref_pump, false);
|
||||
PreferenceManager.setDefaultValues(this, R.xml.pref_rileylink, false);
|
||||
|
||||
setBroadcastReceiver();
|
||||
|
||||
|
||||
|
||||
|
||||
/* start the RoundtripService */
|
||||
/* using startService() will keep the service running until it is explicitly stopped
|
||||
* with stopService() or by RoundtripService calling stopSelf().
|
||||
* Note that calling startService repeatedly has no ill effects on RoundtripService
|
||||
*/
|
||||
// explicitly call startService to keep it running even when the GUI goes away.
|
||||
Intent bindIntent = new Intent(this,RoundtripService.class);
|
||||
startService(bindIntent);
|
||||
|
||||
linearProgressBar = (ProgressBar)findViewById(R.id.progressBarCommandActivity);
|
||||
spinnyProgressBar = (ProgressBar)findViewById(R.id.progressBarSpinny);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void onResume(){
|
||||
super.onResume();
|
||||
|
||||
setBroadcastReceiver();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPause() {
|
||||
super.onPause();
|
||||
if (apsAppConnected != null){
|
||||
LocalBroadcastManager.getInstance(MainApp.instance()).unregisterReceiver(apsAppConnected);
|
||||
}
|
||||
if (mBroadcastReceiver != null){
|
||||
LocalBroadcastManager.getInstance(MainApp.instance()).unregisterReceiver(mBroadcastReceiver);
|
||||
}
|
||||
}
|
||||
|
||||
public void setBroadcastReceiver() {
|
||||
//Register this receiver for UI Updates
|
||||
mBroadcastReceiver = new BroadcastReceiver() {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent receivedIntent) {
|
||||
|
||||
if (receivedIntent == null) {
|
||||
Log.e(TAG, "onReceive: received null intent");
|
||||
} else {
|
||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(MainApp.instance());
|
||||
ServiceTransport transport;
|
||||
|
||||
switch (receivedIntent.getAction()) {
|
||||
case RT2Const.local.INTENT_serviceConnected:
|
||||
case RT2Const.local.INTENT_NEW_rileylinkAddressKey:
|
||||
showIdle();
|
||||
/**
|
||||
* Client MUST send a "UseThisRileylink" message because it asserts that
|
||||
* the user has given explicit permission to use bluetooth.
|
||||
*
|
||||
* We can change the format so that it is a simple "bluetooth OK" message,
|
||||
* rather than an explicit address of a Rileylink, and the Service can
|
||||
* use the last known good value. But the kick-off of bluetooth ops must
|
||||
* come from an Activity.
|
||||
*/
|
||||
String RileylinkBLEAddress = prefs.getString(RT2Const.serviceLocal.rileylinkAddressKey, "");
|
||||
if (RileylinkBLEAddress.equals("")){
|
||||
// TODO: 11/07/2016 @TIM UI message for user
|
||||
Log.e(TAG, "No Rileylink BLE Address saved in app");
|
||||
} else {
|
||||
showBusy("Configuring Service", 50);
|
||||
MainApp.getServiceClientConnection().setThisRileylink(RileylinkBLEAddress);
|
||||
}
|
||||
break;
|
||||
case RT2Const.local.INTENT_NEW_pumpIDKey:
|
||||
MainApp.getServiceClientConnection().sendPUMP_useThisDevice(prefs.getString(RT2Const.serviceLocal.pumpIDKey, ""));
|
||||
break;
|
||||
case RT2Const.local.INTENT_historyPageViewerReady:
|
||||
Intent sendHistoryIntent = new Intent(RT2Const.local.INTENT_historyPageBundleIncoming);
|
||||
sendHistoryIntent.putExtra(RT2Const.IPC.MSG_PUMP_history_key, storeForHistoryViewer);
|
||||
LocalBroadcastManager.getInstance(getApplicationContext()).sendBroadcast(sendHistoryIntent);
|
||||
break;
|
||||
case RT2Const.IPC.MSG_ServiceResult:
|
||||
Log.i(TAG, "Received ServiceResult");
|
||||
|
||||
Bundle bundle = receivedIntent.getBundleExtra(RT2Const.IPC.bundleKey);
|
||||
transport = new ServiceTransport(bundle);
|
||||
if (transport.commandDidCompleteOK()) {
|
||||
String originalCommandName = transport.getOriginalCommandName();
|
||||
switch (originalCommandName) {
|
||||
case "ReadPumpModel":
|
||||
PumpModelResult modelResult = new PumpModelResult();
|
||||
modelResult.initFromServiceResult(transport.getServiceResult());
|
||||
String pumpModelString = modelResult.getPumpModel();
|
||||
// GGW Tue Jul 12 02:29:54 UTC 2016: ok, now what do we do with the pump model?
|
||||
showIdle();
|
||||
break;
|
||||
case "ReadPumpClock":
|
||||
ReadPumpClockResult clockResult = new ReadPumpClockResult();
|
||||
clockResult.initFromServiceResult(transport.getServiceResult());
|
||||
TextView pumpTimeTextView = (TextView) findViewById(R.id.textViewPumpClockTime);
|
||||
pumpTimeTextView.setText(clockResult.getTimeString());
|
||||
showIdle();
|
||||
break;
|
||||
case "FetchPumpHistory":
|
||||
storeForHistoryViewer = receivedIntent.getExtras().getBundle(RT2Const.IPC.bundleKey);
|
||||
startActivity(new Intent(context, HistoryPageListActivity.class));
|
||||
// wait for history viewer to announce "ready"
|
||||
showIdle();
|
||||
break;
|
||||
case "RetrieveHistoryPage":
|
||||
storeForHistoryViewer = receivedIntent.getExtras().getBundle(RT2Const.IPC.bundleKey);
|
||||
startActivity(new Intent(context, HistoryPageListActivity.class));
|
||||
// wait for history viewer to announce "ready"
|
||||
showIdle();
|
||||
break;
|
||||
case "ISFProfile":
|
||||
ISFProfile isfProfile = new ISFProfile();
|
||||
isfProfile.initFromServiceResult(transport.getServiceResult());
|
||||
// TODO: do something with isfProfile
|
||||
showIdle();
|
||||
break;
|
||||
case "BasalProfile":
|
||||
BasalProfile basalProfile = new BasalProfile();
|
||||
basalProfile.initFromServiceResult(transport.getServiceResult());
|
||||
// TODO: do something with basal profile
|
||||
showIdle();
|
||||
break;
|
||||
case "BolusWizardCarbProfile":
|
||||
BolusWizardCarbProfile carbProfile = new BolusWizardCarbProfile();
|
||||
carbProfile.initFromServiceResult(transport.getServiceResult());
|
||||
// TODO: do something with carb profile
|
||||
showIdle();
|
||||
break;
|
||||
case "UpdatePumpStatus":
|
||||
// rebroadcast for HAPP
|
||||
|
||||
break;
|
||||
default:
|
||||
Log.e(TAG, "Dunno what to do with this command completion: " + transport.getOriginalCommandName());
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG,"Command failed? " + transport.getOriginalCommandName());
|
||||
}
|
||||
break;
|
||||
case RT2Const.IPC.MSG_ServiceNotification:
|
||||
transport = new ServiceTransport(receivedIntent.getBundleExtra(RT2Const.IPC.bundleKey));
|
||||
ServiceNotification notification = transport.getServiceNotification();
|
||||
String note = notification.getNotificationType();
|
||||
switch (note) {
|
||||
case RT2Const.IPC.MSG_BLE_RileyLinkReady:
|
||||
setRileylinkStatusMessage("OK");
|
||||
break;
|
||||
case RT2Const.IPC.MSG_PUMP_pumpFound:
|
||||
setPumpStatusMessage("OK");
|
||||
break;
|
||||
case RT2Const.IPC.MSG_PUMP_pumpLost:
|
||||
setPumpStatusMessage("Lost");
|
||||
break;
|
||||
case RT2Const.IPC.MSG_note_WakingPump:
|
||||
showBusy("Waking Pump", 99);
|
||||
break;
|
||||
case RT2Const.IPC.MSG_note_FindingRileyLink:
|
||||
showBusy("Finding RileyLink", 99);
|
||||
break;
|
||||
case RT2Const.IPC.MSG_note_Idle:
|
||||
showIdle();
|
||||
break;
|
||||
default:
|
||||
Log.e(TAG, "Unrecognized Notification: '" + note + "'");
|
||||
}
|
||||
break;
|
||||
default:
|
||||
Log.e(TAG, "Unrecognized intent action: " + receivedIntent.getAction());
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
IntentFilter intentFilter = new IntentFilter();
|
||||
intentFilter.addAction(RT2Const.local.INTENT_serviceConnected);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_ServiceResult);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_ServiceNotification);
|
||||
intentFilter.addAction(RT2Const.local.INTENT_historyPageViewerReady);
|
||||
|
||||
|
||||
linearProgressBar = (ProgressBar)findViewById(R.id.progressBarCommandActivity);
|
||||
spinnyProgressBar = (ProgressBar)findViewById(R.id.progressBarSpinny);
|
||||
LocalBroadcastManager.getInstance(MainApp.instance()).registerReceiver(mBroadcastReceiver, intentFilter);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* GUI element functions
|
||||
*
|
||||
*/
|
||||
|
||||
|
||||
private int mProgress = 0;
|
||||
private int mSpinnyProgress = 0;
|
||||
private ProgressBar linearProgressBar;
|
||||
private ProgressBar spinnyProgressBar;
|
||||
private static final int spinnyFPS = 10;
|
||||
private Thread spinnyThread;
|
||||
void showBusy(String activityString, int progress) {
|
||||
mProgress = progress;
|
||||
TextView tv = (TextView)findViewById(R.id.textViewActivity);
|
||||
tv.setText(activityString);
|
||||
linearProgressBar.setProgress(progress);
|
||||
if (progress > 0) {
|
||||
spinnyProgressBar.setVisibility(View.VISIBLE);
|
||||
if (spinnyThread == null) {
|
||||
spinnyThread = new Thread(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
while ((mProgress > 0) && (mProgress < 100)) {
|
||||
mSpinnyProgress += 100 / spinnyFPS;
|
||||
spinnyProgressBar.setProgress(mSpinnyProgress);
|
||||
SystemClock.sleep(1000 / spinnyFPS);
|
||||
}
|
||||
spinnyThread = null;
|
||||
}
|
||||
});
|
||||
spinnyThread.start();
|
||||
}
|
||||
} else {
|
||||
spinnyProgressBar.setVisibility(View.INVISIBLE);
|
||||
}
|
||||
}
|
||||
|
||||
void showIdle() {
|
||||
showBusy("Idle",0);
|
||||
}
|
||||
|
||||
void setRileylinkStatusMessage(String statusMessage) {
|
||||
TextView field = (TextView)findViewById(R.id.textViewFieldRileyLink);
|
||||
field.setText(statusMessage);
|
||||
}
|
||||
|
||||
void setPumpStatusMessage(String statusMessage) {
|
||||
TextView field = (TextView)findViewById(R.id.textViewFieldPump);
|
||||
field.setText(statusMessage);
|
||||
}
|
||||
|
||||
public void onTunePumpButtonClicked(View view) {
|
||||
MainApp.getServiceClientConnection().doTunePump();
|
||||
}
|
||||
|
||||
public void onFetchHistoryButtonClicked(View view) {
|
||||
/* does not work. Crashes sig 11 */
|
||||
showBusy("Fetch history page 0",50);
|
||||
MainApp.getServiceClientConnection().doFetchPumpHistory();
|
||||
}
|
||||
|
||||
public void onFetchSavedHistoryButtonClicked(View view) {
|
||||
showBusy("Fetching history (not saved)",50);
|
||||
MainApp.getServiceClientConnection().doFetchSavedHistory();
|
||||
}
|
||||
|
||||
public void onReadPumpClockButtonClicked(View view) {
|
||||
showBusy("Reading Pump Clock",50);
|
||||
MainApp.getServiceClientConnection().readPumpClock();
|
||||
}
|
||||
|
||||
public void onGetISFProfileButtonClicked(View view) {
|
||||
//ServiceCommand getISFProfileCommand = ServiceClientActions.makeReadISFProfileCommand();
|
||||
//roundtripServiceClientConnection.sendServiceCommand(getISFProfileCommand);
|
||||
MainApp.getServiceClientConnection().readISFProfile();
|
||||
}
|
||||
|
||||
public void onViewEventLogButtonClicked(View view) {
|
||||
startActivity(new Intent(getApplicationContext(),ServiceMessageViewListActivity.class));
|
||||
}
|
||||
|
||||
public void onUpdateAllStatusButtonClicked(View view) {
|
||||
MainApp.getServiceClientConnection().updateAllStatus();
|
||||
}
|
||||
|
||||
public void onGetCarbProfileButtonClicked(View view) {
|
||||
showBusy("Getting Carb Profile",1);
|
||||
roundtripServiceClientConnection.sendServiceCommand(ServiceClientActions.makeReadBolusWizardCarbProfileCommand());
|
||||
}
|
||||
|
||||
/* UI Setup */
|
||||
@Override
|
||||
public boolean onCreateOptionsMenu(Menu menu) {
|
||||
getMenuInflater().inflate(R.menu.menu_main, menu);
|
||||
return true;
|
||||
}
|
||||
@Override
|
||||
public boolean onKeyUp(int keyCode, KeyEvent event) {
|
||||
switch (keyCode) {
|
||||
case KeyEvent.KEYCODE_MENU:
|
||||
if (mDrawerLayout.isDrawerOpen(GravityCompat.START)){
|
||||
mDrawerLayout.closeDrawers();
|
||||
} else {
|
||||
mDrawerLayout.openDrawer(GravityCompat.START);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
return super.onKeyUp(keyCode, event);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onOptionsItemSelected(MenuItem item) {
|
||||
// Handle item selection
|
||||
switch (item.getItemId()) {
|
||||
case android.R.id.home:
|
||||
mDrawerLayout.openDrawer(mDrawerLinear);
|
||||
return true;
|
||||
|
||||
default:
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
public void setupMenuAndToolbar() {
|
||||
//Setup menu
|
||||
mDrawerLayout = (DrawerLayout)findViewById(R.id.drawer_layout);
|
||||
mDrawerLinear = (LinearLayout) findViewById(R.id.left_drawer);
|
||||
toolbar = (Toolbar) findViewById(R.id.mainActivityToolbar);
|
||||
Drawable logsIcon = getDrawable(R.drawable.file_chart);
|
||||
Drawable historyIcon = getDrawable(R.drawable.history);
|
||||
Drawable settingsIcon = getDrawable(R.drawable.settings);
|
||||
Drawable catIcon = getDrawable(R.drawable.cat);
|
||||
Drawable apsIcon = getDrawable(R.drawable.refresh);
|
||||
|
||||
logsIcon.setColorFilter(getResources().getColor(R.color.primary_dark), PorterDuff.Mode.SRC_ATOP);
|
||||
historyIcon.setColorFilter(getResources().getColor(R.color.primary_dark), PorterDuff.Mode.SRC_ATOP);
|
||||
settingsIcon.setColorFilter(getResources().getColor(R.color.primary_dark), PorterDuff.Mode.SRC_ATOP);
|
||||
catIcon.setColorFilter(getResources().getColor(R.color.primary_dark), PorterDuff.Mode.SRC_ATOP);
|
||||
apsIcon.setColorFilter(getResources().getColor(R.color.primary_dark), PorterDuff.Mode.SRC_ATOP);
|
||||
|
||||
ListView mDrawerList = (ListView)findViewById(R.id.navList);
|
||||
ArrayList<NavItem> menuItems = new ArrayList<>();
|
||||
menuItems.add(new NavItem("APS Integration", apsIcon));
|
||||
menuItems.add(new NavItem("Pump History", historyIcon));
|
||||
menuItems.add(new NavItem("Treatment Logs", logsIcon));
|
||||
menuItems.add(new NavItem("Settings", settingsIcon));
|
||||
menuItems.add(new NavItem("View LogCat", catIcon));
|
||||
DrawerListAdapter adapterMenu = new DrawerListAdapter(this, menuItems);
|
||||
mDrawerList.setAdapter(adapterMenu);
|
||||
mDrawerList.setOnItemClickListener(new AdapterView.OnItemClickListener() {
|
||||
@Override
|
||||
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
|
||||
switch (position) {
|
||||
case 0:
|
||||
//Check APS App Connectivity
|
||||
sendAPSAppMessage(view);
|
||||
break;
|
||||
case 1:
|
||||
//Pump History
|
||||
startActivity(new Intent(getApplicationContext(), HistoryPageListActivity.class));
|
||||
break;
|
||||
case 2:
|
||||
//Treatment Logs
|
||||
startActivity(new Intent(getApplicationContext(), TreatmentHistory.class));
|
||||
break;
|
||||
case 3:
|
||||
//Settings
|
||||
startActivity(new Intent(getApplicationContext(), SettingsActivity.class));
|
||||
break;
|
||||
case 4:
|
||||
//View LogCat
|
||||
tools.showLogging();
|
||||
break;
|
||||
}
|
||||
mDrawerLayout.closeDrawers();
|
||||
}
|
||||
});
|
||||
|
||||
ActionBarDrawerToggle mDrawerToggle = new ActionBarDrawerToggle(this, mDrawerLayout,R.string.drawer_open, R.string.drawer_close) {
|
||||
/** Called when a drawer has settled in a completely open state. */
|
||||
public void onDrawerOpened(View drawerView) {
|
||||
super.onDrawerOpened(drawerView);
|
||||
invalidateOptionsMenu(); // creates call to onPrepareOptionsMenu()
|
||||
|
||||
//Insulin Integration App, try and connect
|
||||
//checkInsulinAppIntegration(false);
|
||||
}
|
||||
/** Called when a drawer has settled in a completely closed state. */
|
||||
public void onDrawerClosed(View view) {
|
||||
super.onDrawerClosed(view);
|
||||
invalidateOptionsMenu(); // creates call to onPrepareOptionsMenu()
|
||||
}
|
||||
};
|
||||
|
||||
setSupportActionBar(toolbar);
|
||||
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
||||
getSupportActionBar().setHomeButtonEnabled(true);
|
||||
mDrawerToggle.syncState();
|
||||
mDrawerToggle.setDrawerIndicatorEnabled(true);
|
||||
mDrawerLayout.addDrawerListener(mDrawerToggle);
|
||||
}
|
||||
|
||||
|
||||
/* Functions for APS App Service */
|
||||
|
||||
//Our Service that APS App will connect to
|
||||
private Messenger myService = null;
|
||||
private ServiceConnection myConnection = new ServiceConnection() {
|
||||
public void onServiceConnected(ComponentName className, IBinder service) {
|
||||
myService = new Messenger(service);
|
||||
|
||||
//Broadcast there has been a connection
|
||||
Intent intent = new Intent("APS_CONNECTED");
|
||||
LocalBroadcastManager.getInstance(MainApp.instance()).sendBroadcast(intent);
|
||||
}
|
||||
|
||||
public void onServiceDisconnected(ComponentName className) {
|
||||
myService = null;
|
||||
//FYI, only called if Service crashed or was killed, not on unbind
|
||||
}
|
||||
};
|
||||
|
||||
public void sendAPSAppMessage(final View view)
|
||||
{
|
||||
//listen out for a successful connection
|
||||
apsAppConnected = new BroadcastReceiver() {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
|
||||
Resources appR = view.getContext().getResources();
|
||||
CharSequence txt = appR.getText(appR.getIdentifier("app_name", "string", view.getContext().getPackageName()));
|
||||
|
||||
Message msg = Message.obtain();
|
||||
Bundle bundle = new Bundle();
|
||||
bundle.putString(RT2Const.commService.ACTION,RT2Const.commService.OUTGOING_TEST_MSG);
|
||||
bundle.putString(RT2Const.commService.REMOTE_APP_NAME, txt.toString());
|
||||
msg.setData(bundle);
|
||||
|
||||
try {
|
||||
myService.send(msg);
|
||||
} catch (RemoteException e) {
|
||||
e.printStackTrace();
|
||||
//cannot Bind to service
|
||||
Snackbar snackbar = Snackbar
|
||||
.make(view, "error sending msg: " + e.getMessage(), Snackbar.LENGTH_INDEFINITE);
|
||||
snackbar.show();
|
||||
}
|
||||
|
||||
if (apsAppConnected != null) LocalBroadcastManager.getInstance(MainApp.instance()).unregisterReceiver(apsAppConnected); //Stop listening for new connections
|
||||
MainApp.instance().unbindService(myConnection);
|
||||
}
|
||||
};
|
||||
LocalBroadcastManager.getInstance(MainApp.instance()).registerReceiver(apsAppConnected, new IntentFilter("APS_CONNECTED"));
|
||||
|
||||
connect_to_aps_app(MainApp.instance());
|
||||
}
|
||||
|
||||
//Connect to the APS App Treatments Service
|
||||
private void connect_to_aps_app(Context c){
|
||||
// TODO: 16/06/2016 add user selected aps app
|
||||
Intent intent = new Intent("com.hypodiabetic.happ.services.TreatmentService");
|
||||
intent.setPackage("com.hypodiabetic.happ");
|
||||
c.bindService(intent, myConnection, Context.BIND_AUTO_CREATE);
|
||||
}
|
||||
|
||||
|
||||
|
||||
}
|
||||
|
||||
class NavItem {
|
||||
String mTitle;
|
||||
Drawable mIcon;
|
||||
|
||||
public NavItem(String title, Drawable icon) {
|
||||
mTitle = title;
|
||||
mIcon = icon;
|
||||
}
|
||||
}
|
||||
|
||||
class DrawerListAdapter extends BaseAdapter {
|
||||
|
||||
Context mContext;
|
||||
ArrayList<NavItem> mNavItems;
|
||||
|
||||
public DrawerListAdapter(Context context, ArrayList<NavItem> navItems) {
|
||||
mContext = context;
|
||||
mNavItems = navItems;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getCount() {
|
||||
return mNavItems.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object getItem(int position) {
|
||||
return mNavItems.get(position);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getItemId(int position) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public View getView(int position, View convertView, ViewGroup parent) {
|
||||
View view;
|
||||
|
||||
if (convertView == null) {
|
||||
LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
|
||||
view = inflater.inflate(R.layout.menu_item, null);
|
||||
}
|
||||
else {
|
||||
view = convertView;
|
||||
}
|
||||
|
||||
TextView titleView = (TextView) view.findViewById(R.id.menuText);
|
||||
ImageView iconView = (ImageView) view.findViewById(R.id.menuIcon);
|
||||
|
||||
titleView.setText( mNavItems.get(position).mTitle);
|
||||
iconView.setBackground(mNavItems.get(position).mIcon);
|
||||
return view;
|
||||
}
|
||||
}
|
|
@ -1,48 +0,0 @@
|
|||
package com.gxwtech.roundtrip2;
|
||||
|
||||
import android.app.Application;
|
||||
|
||||
import io.realm.Realm;
|
||||
import io.realm.RealmConfiguration;
|
||||
|
||||
/**
|
||||
* Created by Tim on 15/06/2016.
|
||||
*/
|
||||
public class MainApp extends Application {
|
||||
|
||||
private static MainApp sInstance;
|
||||
private static ServiceClientConnection serviceClientConnection;
|
||||
|
||||
@Override
|
||||
public void onCreate() {
|
||||
super.onCreate();
|
||||
|
||||
sInstance = this;
|
||||
serviceClientConnection = new ServiceClientConnection();
|
||||
|
||||
//initialize Realm
|
||||
RealmConfiguration realmConfiguration = new RealmConfiguration.Builder(instance())
|
||||
.name("rt2.realm")
|
||||
.schemaVersion(0)
|
||||
.deleteRealmIfMigrationNeeded() // TODO: 03/08/2016 @TIM remove
|
||||
.build();
|
||||
Realm.setDefaultConfiguration(realmConfiguration);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
public static MainApp instance() {
|
||||
return sInstance;
|
||||
}
|
||||
|
||||
public static ServiceClientConnection getServiceClientConnection(){
|
||||
if (serviceClientConnection == null) {
|
||||
serviceClientConnection = new ServiceClientConnection();
|
||||
}
|
||||
return serviceClientConnection;
|
||||
}
|
||||
|
||||
// TODO: 09/07/2016 @TIM uncomment ServiceClientConnection once class is added
|
||||
|
||||
}
|
|
@ -1,135 +0,0 @@
|
|||
package com.gxwtech.roundtrip2;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class RT2Const {
|
||||
public static final String RT2Prefix = "com.gxwtech.roundtrip2.";
|
||||
public class IPC {
|
||||
public static final String Prefix = RT2Prefix + "IPC.";
|
||||
|
||||
// This message is used to bind the "replyTo" field
|
||||
public static final int MSG_clientRegistered = 63; // arbitrary
|
||||
public static final int MSG_unregisterClient = 64;
|
||||
public static final int MSG_registerClient = 65;
|
||||
|
||||
// used in IPC to mark a message as a generic IPC message
|
||||
public static final int MSG_IPC = 66;
|
||||
// used as the key to find the message in the bundle.
|
||||
public static final String messageKey = Prefix + "messageKey";
|
||||
// used as a key for the bundle, when the bundle is packed into an Intent
|
||||
public static final String bundleKey = Prefix + "bundleKey";
|
||||
// a field to hold the instant (milliseconds since 1970) the message
|
||||
// was sent or received by the CLIENT
|
||||
public static final String instantKey = Prefix + "instantKey";
|
||||
// key for the "command" string in a bundle
|
||||
public static final String commandKey = Prefix + "commandKey";
|
||||
// key for the command-response bundle in a result bundle
|
||||
public static final String serviceResultKey = Prefix + "commandResponse";
|
||||
|
||||
// used by gui to pass the address of the Rileylink to the Service
|
||||
// has an 'address' field with the string address.
|
||||
public static final String MSG_BLE_useThisDevice = Prefix + "MSG_BLE_useThisDevice";
|
||||
public static final String MSG_BLE_useThisDevice_addressKey = Prefix + "MSG_BLE_useThisDevice_addressKey";
|
||||
|
||||
// used by gui to tell service that BLE access is denied by user.
|
||||
public static final String MSG_BLE_accessDenied = Prefix + "MSG_BLE_accessDenied";
|
||||
public static final String MSG_BLE_accessGranted = Prefix + "MSG_BLE_accessGranted";
|
||||
// used by service to ask user for Bluetooth permission
|
||||
public static final String MSG_BLE_requestAccess = Prefix + "MSG_BLE_requestAccess";
|
||||
public static final String MSG_BLE_RileyLinkReady = Prefix + "MSG_BLE_RileyLinkReady";
|
||||
|
||||
// used to pass the pump ID from GUI to service
|
||||
// has a 'pumpID' field containing a six digit String
|
||||
public static final String MSG_PUMP_useThisAddress = Prefix + "MSG_PUMP_useThisAddress";
|
||||
public static final String MSG_PUMP_useThisAddress_pumpIDKey = Prefix + "MSG_PUMP_useThisAddress_pumpIDKey";
|
||||
|
||||
// These are used to pass information about the pump from the service to the GUI.
|
||||
// has a 'model' field
|
||||
//public static final String MSG_PUMP_reportedPumpModel = Prefix + "MSG_PUMP_reportedPumpModel";
|
||||
public static final String MSG_PUMP_pumpFound = Prefix + "MSG_PUMP_pumpFound";
|
||||
public static final String MSG_PUMP_pumpLost = Prefix + "MSG_PUMP_pumpLost";
|
||||
|
||||
public static final String MSG_PUMP_tunePump = Prefix + "MSG_PUMP_tunePump";
|
||||
public static final String MSG_PUMP_quickTune = Prefix + "MSG_PUMP_quickTune";
|
||||
public static final String MSG_PUMP_fetchHistory = Prefix + "MSG_PUMP_fetchHistory";
|
||||
|
||||
public static final String MSG_PUMP_history = Prefix + "MSG_PUMP_history";
|
||||
public static final String MSG_PUMP_history_key = Prefix + "MSG_PUMP_history_key";
|
||||
public static final String MSG_PUMP_fetchSavedHistory = Prefix + "MSG_PUMP_fetchSavedHistory";
|
||||
|
||||
// interface for ServiceCommand/ServiceResult
|
||||
public static final String MSG_ServiceCommand = Prefix + "MSG_ServiceCommand";
|
||||
public static final String MSG_ServiceResult = Prefix + "MSG_ServiceResult";
|
||||
public static final String MSG_ServiceNotification = Prefix + "MSG_ServiceNotification";
|
||||
|
||||
// These are notifications to GUIs to let them know what's happening
|
||||
public static final String MSG_note_Idle = Prefix + "MSG_note_Idle";
|
||||
public static final String MSG_note_FindingRileyLink = Prefix + "MSG_note_FindingRileyLink";
|
||||
public static final String MSG_note_WakingPump = Prefix + "MSG_note_WakingPump";
|
||||
public static final String MSG_note_TaskProgress = Prefix + "MSG_note_TaskProgress";
|
||||
|
||||
}
|
||||
public class local {
|
||||
// These are local to the GUI activities
|
||||
public static final String Prefix = RT2Prefix + "local.";
|
||||
public static final String INTENT_serviceConnected = Prefix + "INTENT_serviceConnected";
|
||||
public static final String INTENT_NEW_rileylinkAddressKey = Prefix + "INTENT_NEW_rileylinkAddressKey";
|
||||
public static final String INTENT_NEW_pumpIDKey = Prefix + "INTENT_NEW_pumpIDKey";
|
||||
|
||||
|
||||
public static final String INTENT_historyPageViewerReady = Prefix + "I'm ready, hit me up";
|
||||
public static final String INTENT_historyPageBundleIncoming = Prefix + "Here's the kitchen sink";
|
||||
}
|
||||
|
||||
public class serviceLocal {
|
||||
public static final String Prefix = RT2Prefix + "serviceLocal.";
|
||||
// for local broadcasts annoucing connectivity events
|
||||
public static final String INTENT_seekRileylink = Prefix + "INTENT_seekRileylink";
|
||||
public static final String bluetooth_connected = Prefix + "bluetooth_connected";
|
||||
public static final String bluetooth_disconnected = Prefix + "bluetooth_disconnected";
|
||||
public static final String BLE_services_discovered = Prefix + "BLE_services_discovered";
|
||||
|
||||
public static final String ipcBound = Prefix + "ipcBound";
|
||||
|
||||
// primary shared preferences file identifier
|
||||
public static final String sharedPreferencesKey = Prefix + "sharedPreferencesKey";
|
||||
|
||||
// These are used to identify shared preference items
|
||||
public static final String pumpIDKey = Prefix + "PumpIDKey";
|
||||
public static final String rileylinkAddressKey = Prefix + "rileylinkAddressKey";
|
||||
public static final String prefsLastGoodPumpFrequency = Prefix + "prefsLastGoodPumpFrequency";
|
||||
|
||||
// The the key to identify the hashCode() of the msg.replyTo, when the bundle is moved to an intent.
|
||||
public static final String IPCReplyTo_hashCodeKey = Prefix + "IPCReplyTo_hashCodeKey";
|
||||
|
||||
// This is sent from the PumpManager to RoundtripService at the completion of a pump command session.
|
||||
public static final String INTENT_sessionCompleted = Prefix + "INTENT_sessionCompleted";
|
||||
|
||||
}
|
||||
|
||||
public class commService {
|
||||
//Data
|
||||
public static final String ACTION = "ACTION";
|
||||
public static final String DATE_REQUESTED = "DATE_REQUESTED";
|
||||
public static final String INTEGRATION_OBJECTS = "INTEGRATION_OBJECTS";
|
||||
public static final String TREATMENT_OBJECTS = "TREATMENT_OBJECTS";
|
||||
public static final String PUMP = "PUMP";
|
||||
public static final String REMOTE_APP_NAME = "REMOTE_APP_NAME";
|
||||
|
||||
//Incoming actions
|
||||
public static final String INCOMING_NEW_TREATMENTS = "NEW_TREATMENTS";
|
||||
public static final String INCOMING_TEST_MSG = "TEST_MSG";
|
||||
|
||||
//Outgoing actions
|
||||
public static final String OUTGOING_TREATMENT_UPDATES = "TREATMENT_UPDATES";
|
||||
public static final String OUTGOING_TEST_MSG = "TEST_MSG";
|
||||
|
||||
}
|
||||
|
||||
public class safety {
|
||||
public static final int INCOMING_REQUEST_MAX_AGE = 10; //mins
|
||||
public static final int TREATMENT_MAX_AGE = 8; //mins
|
||||
|
||||
}
|
||||
}
|
|
@ -1,343 +0,0 @@
|
|||
package com.gxwtech.roundtrip2;
|
||||
|
||||
import android.Manifest;
|
||||
import android.app.AlertDialog;
|
||||
import android.bluetooth.BluetoothAdapter;
|
||||
import android.bluetooth.BluetoothDevice;
|
||||
import android.bluetooth.BluetoothManager;
|
||||
import android.bluetooth.le.BluetoothLeScanner;
|
||||
import android.bluetooth.le.ScanFilter;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.os.Build;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.support.design.widget.Snackbar;
|
||||
import android.support.v4.app.ActivityCompat;
|
||||
import android.support.v4.content.ContextCompat;
|
||||
import android.support.v4.content.LocalBroadcastManager;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.view.Menu;
|
||||
import android.view.MenuItem;
|
||||
import android.view.View;
|
||||
import android.widget.AdapterView;
|
||||
import android.widget.Toast;
|
||||
import android.util.Log;
|
||||
|
||||
import android.bluetooth.le.ScanCallback;
|
||||
import android.bluetooth.le.ScanResult;
|
||||
import android.bluetooth.le.ScanSettings;
|
||||
|
||||
import com.gxwtech.roundtrip2.util.LocationHelper;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import android.content.SharedPreferences;
|
||||
import android.preference.PreferenceManager;
|
||||
import android.view.LayoutInflater;
|
||||
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.BaseAdapter;
|
||||
import android.widget.ListView;
|
||||
import android.widget.TextView;
|
||||
|
||||
|
||||
public class RileyLinkScan extends AppCompatActivity {
|
||||
private final static String TAG = "RileyLinkScan";
|
||||
private BluetoothAdapter mBluetoothAdapter;
|
||||
private BluetoothLeScanner mLEScanner;
|
||||
private LeDeviceListAdapter mLeDeviceListAdapter;
|
||||
public boolean mScanning;
|
||||
private Handler mHandler;
|
||||
public Snackbar snackbar;
|
||||
public ScanSettings settings;
|
||||
public List<ScanFilter> filters;
|
||||
public ListView listBTScan;
|
||||
public Toolbar toolbarBTScan;
|
||||
public Context mContext = this;
|
||||
|
||||
private static final int PERMISSION_REQUEST_COARSE_LOCATION = 30241; // arbitrary.
|
||||
private static final int REQUEST_ENABLE_BT = 30242; // arbitrary
|
||||
// Stops scanning after 10 seconds.
|
||||
private static final long SCAN_PERIOD = 10000;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_riley_link_scan);
|
||||
|
||||
// Initializes Bluetooth adapter.
|
||||
final BluetoothManager bluetoothManager = (BluetoothManager) getSystemService(Context.BLUETOOTH_SERVICE);
|
||||
mBluetoothAdapter = bluetoothManager.getAdapter();
|
||||
mHandler = new Handler();
|
||||
|
||||
mLeDeviceListAdapter = new LeDeviceListAdapter();
|
||||
listBTScan = (ListView) findViewById(R.id.listBTScan);
|
||||
listBTScan.setAdapter(mLeDeviceListAdapter);
|
||||
listBTScan.setOnItemClickListener(new AdapterView.OnItemClickListener() {
|
||||
@Override
|
||||
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
|
||||
TextView textview = (TextView) view.findViewById(R.id.device_address);
|
||||
String bleAddress = textview.getText().toString();
|
||||
|
||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
|
||||
prefs.edit().putString(RT2Const.serviceLocal.rileylinkAddressKey, bleAddress).apply();
|
||||
|
||||
//Notify that we have a new rileylinkAddressKey
|
||||
LocalBroadcastManager.getInstance(MainApp.instance()).sendBroadcast(new Intent(RT2Const.local.INTENT_NEW_rileylinkAddressKey));
|
||||
|
||||
Log.d(TAG, "New rileylinkAddressKey: " + bleAddress);
|
||||
|
||||
//Notify that we have a new pumpIDKey
|
||||
LocalBroadcastManager.getInstance(MainApp.instance()).sendBroadcast(new Intent(RT2Const.local.INTENT_NEW_pumpIDKey));
|
||||
finish();
|
||||
}
|
||||
});
|
||||
|
||||
toolbarBTScan = (Toolbar) findViewById(R.id.toolbarBTScan);
|
||||
toolbarBTScan.setTitle(R.string.title_activity_riley_link_scan);
|
||||
setSupportActionBar(toolbarBTScan);
|
||||
|
||||
snackbar = Snackbar.make(findViewById(R.id.RileyLinkScan), "Scanning...",Snackbar.LENGTH_INDEFINITE);
|
||||
snackbar.setAction("STOP", new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
scanLeDevice(false);
|
||||
}
|
||||
});
|
||||
|
||||
startScanBLE();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPause() {
|
||||
super.onPause();
|
||||
scanLeDevice(false);
|
||||
mLeDeviceListAdapter.clear();
|
||||
mLeDeviceListAdapter.notifyDataSetChanged();
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public boolean onCreateOptionsMenu(Menu menu) {
|
||||
getMenuInflater().inflate(R.menu.menu_bluetooth_scan, menu);
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onOptionsItemSelected(MenuItem item) {
|
||||
switch (item.getItemId()) {
|
||||
case R.id.miScan:
|
||||
startScanBLE();
|
||||
return true;
|
||||
default:
|
||||
return super.onOptionsItemSelected(item);
|
||||
}
|
||||
}
|
||||
|
||||
public void startScanBLE(){
|
||||
// https://developer.android.com/training/permissions/requesting.html
|
||||
// http://developer.radiusnetworks.com/2015/09/29/is-your-beacon-app-ready-for-android-6.html
|
||||
if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE)) {
|
||||
Toast.makeText(this, "R.string.ble_not_supported", Toast.LENGTH_SHORT).show();
|
||||
} else {
|
||||
// Use this check to determine whether BLE is supported on the device. Then
|
||||
// you can selectively disable BLE-related features.
|
||||
if (ContextCompat.checkSelfPermission(mContext, Manifest.permission.ACCESS_COARSE_LOCATION)
|
||||
!= PackageManager.PERMISSION_GRANTED) {
|
||||
//your code that requires permission
|
||||
ActivityCompat.requestPermissions(this, new String[] { Manifest.permission.ACCESS_COARSE_LOCATION },
|
||||
PERMISSION_REQUEST_COARSE_LOCATION);
|
||||
}
|
||||
|
||||
// Ensures Bluetooth is available on the device and it is enabled. If not,
|
||||
// displays a dialog requesting user permission to enable Bluetooth.
|
||||
if (mBluetoothAdapter == null || !mBluetoothAdapter.isEnabled()) {
|
||||
Toast.makeText(this, "R.string.ble_not_enabled", Toast.LENGTH_SHORT).show();
|
||||
} else {
|
||||
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
||||
// Will request that GPS be enabled for devices running Marshmallow or newer.
|
||||
LocationHelper.requestLocationForBluetooth(this);
|
||||
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
|
||||
Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
|
||||
startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT);
|
||||
}
|
||||
|
||||
mLEScanner = mBluetoothAdapter.getBluetoothLeScanner();
|
||||
settings = new ScanSettings.Builder()
|
||||
.setScanMode(ScanSettings.SCAN_MODE_LOW_LATENCY)
|
||||
.build();
|
||||
filters = new ArrayList<ScanFilter>();
|
||||
|
||||
scanLeDevice(true);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
||||
super.onActivityResult(requestCode, resultCode, data);
|
||||
if (requestCode == REQUEST_ENABLE_BT) {
|
||||
if (resultCode == RESULT_OK) {
|
||||
// User allowed Bluetooth to turn on
|
||||
} else if (resultCode == RESULT_CANCELED) {
|
||||
// Error, or user said "NO"
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void scanLeDevice(final boolean enable) {
|
||||
if (enable) {
|
||||
// Stops scanning after a pre-defined scan period.
|
||||
mHandler.postDelayed(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
mScanning = false;
|
||||
mLEScanner.stopScan(mScanCallback);
|
||||
Log.d(TAG, "scanLeDevice: Scanning Stop");
|
||||
//Toast.makeText(mContext, "Scanning finished", Toast.LENGTH_SHORT).show();
|
||||
snackbar.dismiss();
|
||||
}
|
||||
}, SCAN_PERIOD);
|
||||
|
||||
mScanning = true;
|
||||
mLEScanner.startScan(mScanCallback);
|
||||
Log.d(TAG, "scanLeDevice: Scanning Start");
|
||||
//Toast.makeText(this, "Scanning", Toast.LENGTH_SHORT).show();
|
||||
snackbar.show();
|
||||
} else {
|
||||
mScanning = false;
|
||||
mLEScanner.stopScan(mScanCallback);
|
||||
Log.d(TAG, "scanLeDevice: Scanning Stop");
|
||||
//Toast.makeText(this, "Scanning finished", Toast.LENGTH_SHORT).show();
|
||||
snackbar.dismiss();
|
||||
|
||||
}
|
||||
}
|
||||
private ScanCallback mScanCallback = new ScanCallback() {
|
||||
@Override
|
||||
public void onScanResult(int callbackType, ScanResult result) {
|
||||
final BluetoothDevice device = result.getDevice();
|
||||
runOnUiThread(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
if (device.getName() != null && device.getName().length() > 0) {
|
||||
mLeDeviceListAdapter.addDevice(device);
|
||||
mLeDeviceListAdapter.notifyDataSetChanged();
|
||||
Log.d(TAG, "Found BLE" + device.getName());
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBatchScanResults(final List<ScanResult> results) {
|
||||
runOnUiThread(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
for (ScanResult result: results) {
|
||||
BluetoothDevice device = result.getDevice();
|
||||
if (device.getName() != null && device.getName().length() > 0) {
|
||||
mLeDeviceListAdapter.addDevice(device);
|
||||
Log.d(TAG, "Found BLE" + result.toString());
|
||||
} else {
|
||||
Log.e(TAG, "Found BLE, but name appears to be missing. Ignoring. " + device.getAddress());
|
||||
}
|
||||
}
|
||||
mLeDeviceListAdapter.notifyDataSetChanged();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onScanFailed(int errorCode) {
|
||||
Log.e("Scan Failed", "Error Code: " + errorCode);
|
||||
Toast.makeText(mContext, "Scan Failed " + errorCode, Toast.LENGTH_LONG).show();
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
|
||||
private class LeDeviceListAdapter extends BaseAdapter {
|
||||
private ArrayList<BluetoothDevice> mLeDevices;
|
||||
private LayoutInflater mInflator;
|
||||
|
||||
public LeDeviceListAdapter() {
|
||||
super();
|
||||
mLeDevices = new ArrayList<>();
|
||||
mInflator = RileyLinkScan.this.getLayoutInflater();
|
||||
}
|
||||
|
||||
public void addDevice(BluetoothDevice device) {
|
||||
if(!mLeDevices.contains(device)) {
|
||||
mLeDevices.add(device);
|
||||
notifyDataSetChanged();
|
||||
}
|
||||
}
|
||||
|
||||
public BluetoothDevice getDevice(int position) {
|
||||
return mLeDevices.get(position);
|
||||
}
|
||||
|
||||
public void clear() {
|
||||
mLeDevices.clear();
|
||||
notifyDataSetChanged();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getCount() {
|
||||
return mLeDevices.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object getItem(int i) {
|
||||
return mLeDevices.get(i);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getItemId(int i) {
|
||||
return i;
|
||||
}
|
||||
|
||||
@Override
|
||||
public View getView(int i, View view, ViewGroup viewGroup) {
|
||||
ViewHolder viewHolder;
|
||||
// General ListView optimization code.
|
||||
if (view == null) {
|
||||
view = mInflator.inflate(R.layout.listitem_device, null);
|
||||
viewHolder = new ViewHolder();
|
||||
viewHolder.deviceAddress = (TextView) view.findViewById(R.id.device_address);
|
||||
viewHolder.deviceName = (TextView) view.findViewById(R.id.device_name);
|
||||
view.setTag(viewHolder);
|
||||
} else {
|
||||
viewHolder = (ViewHolder) view.getTag();
|
||||
}
|
||||
|
||||
BluetoothDevice device = mLeDevices.get(i);
|
||||
String deviceName = device.getName();
|
||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
|
||||
if(prefs.getString(RT2Const.serviceLocal.rileylinkAddressKey, "").compareTo(device.getAddress()) == 0) {
|
||||
viewHolder.deviceName.setTextColor(getColor(R.color.secondary_text_light));
|
||||
viewHolder.deviceAddress.setTextColor(getColor(R.color.secondary_text_light));
|
||||
deviceName += " (" + getResources().getString(R.string.selected_device) + ")";
|
||||
}
|
||||
viewHolder.deviceName.setText(deviceName);
|
||||
viewHolder.deviceAddress.setText(device.getAddress());
|
||||
|
||||
return view;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static class ViewHolder {
|
||||
TextView deviceName;
|
||||
TextView deviceAddress;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,21 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLink;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
import java.util.Comparator;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/30/16.
|
||||
*/
|
||||
public class FrequencyScanResults {
|
||||
public ArrayList<FrequencyTrial> trials = new ArrayList<>();
|
||||
public double bestFrequencyMHz = 0.0;
|
||||
public void sort() {
|
||||
Collections.sort(trials, new Comparator<FrequencyTrial>() {
|
||||
@Override
|
||||
public int compare(FrequencyTrial trial1, FrequencyTrial trial2) {
|
||||
return trial1.averageRSSI.compareTo(trial2.averageRSSI);
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
|
@ -1,11 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLink;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/30/16.
|
||||
*/
|
||||
public class FrequencyTrial {
|
||||
public int tries = 0;
|
||||
public int successes = 0;
|
||||
public Double averageRSSI = 0.0;
|
||||
public double frequencyMHz = 0.0;
|
||||
}
|
|
@ -1,599 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLink;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.SharedPreferences;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RFSpy;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RFSpyResponse;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RadioPacket;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RadioResponse;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages.ButtonPressCarelinkMessageBody;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages.CarelinkShortMessageBody;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages.GetHistoryPageCarelinkMessageBody;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages.GetPumpModelCarelinkMessageBody;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages.MessageBody;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages.MessageType;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages.PumpAckMessageBody;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PacketType;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.ISFTable;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.Page;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.Record;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpMessage;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ReadPumpClockResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.StringUtil;
|
||||
|
||||
import org.joda.time.Duration;
|
||||
import org.joda.time.IllegalFieldValueException;
|
||||
import org.joda.time.Instant;
|
||||
import org.joda.time.LocalDateTime;
|
||||
|
||||
import java.util.ArrayList;
|
||||
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/30/16.
|
||||
*/
|
||||
public class PumpManager {
|
||||
private static final String TAG = "PumpManager";
|
||||
public double[] scanFrequencies = {916.45, 916.50, 916.55, 916.60, 916.65, 916.70, 916.75, 916.80};
|
||||
public static final int startSession_signal = 6656; // arbitrary.
|
||||
//private long pumpAwakeUntil = 0;
|
||||
private int pumpAwakeForMinutes = 6;
|
||||
private final RFSpy rfspy;
|
||||
private byte[] pumpID;
|
||||
public boolean DEBUG_PUMPMANAGER = true;
|
||||
private final Context context;
|
||||
private SharedPreferences prefs;
|
||||
private Instant lastGoodPumpCommunicationTime = new Instant(0);
|
||||
|
||||
public PumpManager(Context context, RFSpy rfspy, byte[] pumpID) {
|
||||
this.context = context;
|
||||
this.rfspy = rfspy;
|
||||
this.pumpID = pumpID;
|
||||
prefs = context.getSharedPreferences(RT2Const.serviceLocal.sharedPreferencesKey, Context.MODE_PRIVATE);
|
||||
}
|
||||
|
||||
private PumpMessage runCommandWithArgs(PumpMessage msg) {
|
||||
PumpMessage rval;
|
||||
PumpMessage shortMessage = makePumpMessage(msg.messageType, new CarelinkShortMessageBody(new byte[]{0}));
|
||||
// look for ack from short message
|
||||
PumpMessage shortResponse = sendAndListen(shortMessage);
|
||||
if (shortResponse.messageType.mtype == MessageType.PumpAck) {
|
||||
rval = sendAndListen(msg);
|
||||
return rval;
|
||||
} else {
|
||||
Log.e(TAG, "runCommandWithArgs: Pump did not ack Attention packet");
|
||||
}
|
||||
return new PumpMessage();
|
||||
}
|
||||
|
||||
protected PumpMessage sendAndListen(PumpMessage msg) {
|
||||
return sendAndListen(msg, 2000);
|
||||
}
|
||||
|
||||
// All pump communications go through this function.
|
||||
protected PumpMessage sendAndListen(PumpMessage msg, int timeout_ms) {
|
||||
boolean showPumpMessages = true;
|
||||
if (showPumpMessages) {
|
||||
Log.i(TAG, "Sent:" + ByteUtil.shortHexString(msg.getTxData()));
|
||||
}
|
||||
RFSpyResponse resp = rfspy.transmitThenReceive(new RadioPacket(msg.getTxData()), timeout_ms);
|
||||
PumpMessage rval = new PumpMessage(resp.getRadioResponse().getPayload());
|
||||
if (rval.isValid()) {
|
||||
// Mark this as the last time we heard from the pump.
|
||||
rememberLastGoodPumpCommunicationTime();
|
||||
}
|
||||
if (showPumpMessages) {
|
||||
Log.i(TAG, "Received:" + ByteUtil.shortHexString(resp.getRadioResponse().getPayload()));
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
public Page getPumpHistoryPage(int pageNumber) {
|
||||
RawHistoryPage rval = new RawHistoryPage();
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
PumpMessage getHistoryMsg = makePumpMessage(new MessageType(MessageType.CMD_M_READ_HISTORY), new GetHistoryPageCarelinkMessageBody(pageNumber));
|
||||
//Log.i(TAG,"getPumpHistoryPage("+pageNumber+"): "+ByteUtil.shortHexString(getHistoryMsg.getTxData()));
|
||||
// Ask the pump to transfer history (we get first frame?)
|
||||
PumpMessage firstResponse = runCommandWithArgs(getHistoryMsg);
|
||||
//Log.i(TAG,"getPumpHistoryPage("+pageNumber+"): " + ByteUtil.shortHexString(firstResponse.getContents()));
|
||||
|
||||
PumpMessage ackMsg = makePumpMessage(MessageType.PumpAck, new PumpAckMessageBody());
|
||||
GetHistoryPageCarelinkMessageBody currentResponse = new GetHistoryPageCarelinkMessageBody(firstResponse.getMessageBody().getTxData());
|
||||
int expectedFrameNum = 1;
|
||||
boolean done = false;
|
||||
//while (expectedFrameNum == currentResponse.getFrameNumber()) {
|
||||
int failures = 0;
|
||||
while (!done) {
|
||||
// examine current response for problems.
|
||||
byte[] frameData = currentResponse.getFrameData();
|
||||
if ((frameData != null) && (frameData.length > 0) && currentResponse.getFrameNumber() == expectedFrameNum) {
|
||||
// success! got a frame.
|
||||
if (frameData.length != 64) {
|
||||
Log.w(TAG, "Expected frame of length 64, got frame of length " + frameData.length);
|
||||
// but append it anyway?
|
||||
}
|
||||
// handle successful frame data
|
||||
rval.appendData(currentResponse.getFrameData());
|
||||
RoundtripService.getInstance().announceProgress(((100 / 16) * currentResponse.getFrameNumber() + 1));
|
||||
Log.i(TAG, "getPumpHistoryPage: Got frame " + currentResponse.getFrameNumber());
|
||||
// Do we need to ask for the next frame?
|
||||
if (expectedFrameNum < 16) { // This number may not be correct for pumps other than 522/722
|
||||
expectedFrameNum++;
|
||||
} else {
|
||||
done = true; // successful completion
|
||||
}
|
||||
} else {
|
||||
if (frameData == null) {
|
||||
Log.e(TAG, "null frame data, retrying");
|
||||
} else if (currentResponse.getFrameNumber() != expectedFrameNum) {
|
||||
Log.w(TAG, String.format("Expected frame number %d, received %d (retrying)", expectedFrameNum, currentResponse.getFrameNumber()));
|
||||
} else if (frameData.length == 0) {
|
||||
Log.w(TAG, "Frame has zero length, retrying");
|
||||
}
|
||||
failures++;
|
||||
if (failures == 6) {
|
||||
Log.e(TAG, String.format("6 failures in attempting to download frame %d of page %d, giving up.", expectedFrameNum, pageNumber));
|
||||
done = true; // failure completion.
|
||||
}
|
||||
}
|
||||
if (!done) {
|
||||
// ask for next frame
|
||||
PumpMessage nextMsg = sendAndListen(ackMsg);
|
||||
currentResponse = new GetHistoryPageCarelinkMessageBody(nextMsg.getMessageBody().getTxData());
|
||||
}
|
||||
}
|
||||
if (rval.getLength() != 1024) {
|
||||
Log.w(TAG, "getPumpHistoryPage: short page. Expected length of 1024, found length of " + rval.getLength());
|
||||
}
|
||||
if (rval.isChecksumOK() == false) {
|
||||
Log.e(TAG, "getPumpHistoryPage: checksum is wrong");
|
||||
}
|
||||
|
||||
rval.dumpToDebug();
|
||||
|
||||
Page page = new Page();
|
||||
//page.parseFrom(rval.getData(),PumpModel.MM522);
|
||||
page.parseFrom(rval.getData(), PumpModel.MM522);
|
||||
|
||||
return page;
|
||||
}
|
||||
|
||||
public ArrayList<Page> getAllHistoryPages() {
|
||||
ArrayList<Page> pages = new ArrayList<>();
|
||||
|
||||
for (int pageNum = 0; pageNum < 16; pageNum++) {
|
||||
pages.add(getPumpHistoryPage(pageNum));
|
||||
}
|
||||
|
||||
return pages;
|
||||
}
|
||||
|
||||
public ArrayList<Page> getHistoryEventsSinceDate(Instant when) {
|
||||
ArrayList<Page> pages = new ArrayList<>();
|
||||
for (int pageNum = 0; pageNum < 16; pageNum++) {
|
||||
pages.add(getPumpHistoryPage(pageNum));
|
||||
for (Page page : pages) {
|
||||
for (Record r : page.mRecordList) {
|
||||
LocalDateTime timestamp = r.getTimestamp().getLocalDateTime();
|
||||
Log.i(TAG, "Found record: (" + r.getClass().getSimpleName() + ") " + timestamp.toString());
|
||||
}
|
||||
}
|
||||
}
|
||||
return pages;
|
||||
}
|
||||
|
||||
private LocalDateTime parsePumpRTCBytes(byte[] bytes) {
|
||||
if (bytes == null) return null;
|
||||
if (bytes.length < 7) return null;
|
||||
int hours = ByteUtil.asUINT8(bytes[0]);
|
||||
int minutes = ByteUtil.asUINT8(bytes[1]);
|
||||
int seconds = ByteUtil.asUINT8(bytes[2]);
|
||||
int year = (ByteUtil.asUINT8(bytes[4]) & 0x3f) + 1984;
|
||||
int month = ByteUtil.asUINT8(bytes[5]);
|
||||
int day = ByteUtil.asUINT8(bytes[6]);
|
||||
try {
|
||||
LocalDateTime pumpTime = new LocalDateTime(year, month, day, hours, minutes, seconds);
|
||||
return pumpTime;
|
||||
} catch (IllegalFieldValueException e) {
|
||||
Log.e(TAG, String.format("parsePumpRTCBytes: Failed to parse pump time value: year=%d, month=%d, hours=%d, minutes=%d, seconds=%d", year, month, day, hours, minutes, seconds));
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
public ReadPumpClockResult getPumpRTC() {
|
||||
ReadPumpClockResult rval = new ReadPumpClockResult();
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
PumpMessage getRTCMsg = makePumpMessage(new MessageType(MessageType.CMD_M_READ_RTC), new CarelinkShortMessageBody(new byte[]{0}));
|
||||
Log.i(TAG, "getPumpRTC: " + ByteUtil.shortHexString(getRTCMsg.getTxData()));
|
||||
PumpMessage response = sendAndListen(getRTCMsg);
|
||||
if (response.isValid()) {
|
||||
byte[] receivedData = response.getContents();
|
||||
if (receivedData != null) {
|
||||
if (receivedData.length >= 9) {
|
||||
LocalDateTime pumpTime = parsePumpRTCBytes(ByteUtil.substring(receivedData, 2, 7));
|
||||
if (pumpTime != null) {
|
||||
rval.setTime(pumpTime);
|
||||
rval.setResultOK();
|
||||
} else {
|
||||
rval.setResultError(ServiceResult.ERROR_MALFORMED_PUMP_RESPONSE);
|
||||
}
|
||||
} else {
|
||||
rval.setResultError(ServiceResult.ERROR_MALFORMED_PUMP_RESPONSE);
|
||||
}
|
||||
} else {
|
||||
rval.setResultError(ServiceResult.ERROR_MALFORMED_PUMP_RESPONSE);
|
||||
}
|
||||
} else {
|
||||
rval.setResultError(ServiceResult.ERROR_INVALID_PUMP_RESPONSE);
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
public PumpModel getPumpModel() {
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
PumpMessage msg = makePumpMessage(new MessageType(MessageType.GetPumpModel), new GetPumpModelCarelinkMessageBody());
|
||||
Log.i(TAG, "getPumpModel: " + ByteUtil.shortHexString(msg.getTxData()));
|
||||
PumpMessage response = sendAndListen(msg);
|
||||
Log.i(TAG, "getPumpModel response: " + ByteUtil.shortHexString(response.getContents()));
|
||||
byte[] contents = response.getContents();
|
||||
PumpModel rval = PumpModel.UNSET;
|
||||
if (contents != null) {
|
||||
if (contents.length >= 7) {
|
||||
rval = PumpModel.fromString(StringUtil.fromBytes(ByteUtil.substring(contents, 3, 3)));
|
||||
} else {
|
||||
Log.w(TAG, "getPumpModel: Cannot return pump model number: data is too short.");
|
||||
}
|
||||
} else {
|
||||
Log.w(TAG, "getPumpModel: Cannot return pump model number: null response");
|
||||
}
|
||||
|
||||
return rval;
|
||||
}
|
||||
|
||||
|
||||
public ISFTable getPumpISFProfile() {
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
PumpMessage getISFProfileMessage = makePumpMessage(new MessageType(MessageType.GetISFProfile), new CarelinkShortMessageBody());
|
||||
PumpMessage resp = sendAndListen(getISFProfileMessage);
|
||||
ISFTable table = new ISFTable();
|
||||
table.parseFrom(resp.getContents());
|
||||
return table;
|
||||
}
|
||||
|
||||
public PumpMessage getBolusWizardCarbProfile() {
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
PumpMessage getCarbProfileMessage = makePumpMessage(new MessageType(MessageType.CMD_M_READ_CARB_RATIOS), new CarelinkShortMessageBody());
|
||||
PumpMessage resp = sendAndListen(getCarbProfileMessage);
|
||||
return resp;
|
||||
}
|
||||
|
||||
public void tryoutPacket(byte[] pkt) {
|
||||
sendAndListen(makePumpMessage(pkt));
|
||||
}
|
||||
|
||||
public void hunt() {
|
||||
tryoutPacket(new byte[]{MessageType.CMD_M_READ_PUMP_STATUS, 0});
|
||||
tryoutPacket(new byte[]{MessageType.CMD_M_READ_FIRMWARE_VER, 0});
|
||||
tryoutPacket(new byte[]{MessageType.CMD_M_READ_INSULIN_REMAINING, 0});
|
||||
|
||||
}
|
||||
|
||||
// See ButtonPressCarelinkMessageBody
|
||||
public void pressButton(int which) {
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
PumpMessage pressButtonMessage = makePumpMessage(new MessageType(MessageType.CMD_M_KEYPAD_PUSH), new ButtonPressCarelinkMessageBody(which));
|
||||
PumpMessage resp = sendAndListen(pressButtonMessage);
|
||||
if (resp.messageType.mtype != MessageType.PumpAck) {
|
||||
Log.e(TAG, "Pump did not ack button press.");
|
||||
}
|
||||
}
|
||||
|
||||
public void wakeup(int duration_minutes) {
|
||||
// If it has been longer than n minutes, do wakeup. Otherwise assume pump is still awake.
|
||||
// **** FIXME: this wakeup doesn't seem to work well... must revisit
|
||||
pumpAwakeForMinutes = duration_minutes;
|
||||
Instant lastGood = getLastGoodPumpCommunicationTime();
|
||||
// Instant lastGoodPlus = lastGood.plus(new Duration(receiverDeviceAwakeForMinutes * 60 * 1000));
|
||||
Instant lastGoodPlus = lastGood.plus(new Duration(1 * 60 * 1000));
|
||||
Instant now = Instant.now();
|
||||
if (now.compareTo(lastGoodPlus) > 0) {
|
||||
Log.i(TAG, "Waking pump...");
|
||||
PumpMessage msg = makePumpMessage(new MessageType(MessageType.PowerOn), new CarelinkShortMessageBody(new byte[]{(byte) duration_minutes}));
|
||||
RFSpyResponse resp = rfspy.transmitThenReceive(new RadioPacket(msg.getTxData()), (byte) 0, (byte) 200, (byte) 0, (byte) 0, 15000, (byte) 0);
|
||||
Log.i(TAG, "wakeup: raw response is " + ByteUtil.shortHexString(resp.getRaw()));
|
||||
} else {
|
||||
Log.v(TAG, "Last pump communication was recent, not waking pump.");
|
||||
}
|
||||
}
|
||||
|
||||
public void setRadioFrequencyForPump(double freqMHz) {
|
||||
rfspy.setBaseFrequency(freqMHz);
|
||||
}
|
||||
|
||||
public double tuneForPump() {
|
||||
return scanForPump(scanFrequencies);
|
||||
}
|
||||
|
||||
private int tune_tryFrequency(double freqMHz) {
|
||||
rfspy.setBaseFrequency(freqMHz);
|
||||
PumpMessage msg = makePumpMessage(new MessageType(MessageType.GetPumpModel), new GetPumpModelCarelinkMessageBody());
|
||||
RadioPacket pkt = new RadioPacket(msg.getTxData());
|
||||
RFSpyResponse resp = rfspy.transmitThenReceive(pkt, (byte) 0, (byte) 0, (byte) 0, (byte) 0, rfspy.EXPECTED_MAX_BLUETOOTH_LATENCY_MS, (byte) 0);
|
||||
if (resp.wasTimeout()) {
|
||||
Log.w(TAG, String.format("tune_tryFrequency: no pump response at frequency %.2f", freqMHz));
|
||||
} else if (resp.looksLikeRadioPacket()) {
|
||||
RadioResponse radioResponse = new RadioResponse(resp.getRaw());
|
||||
if (radioResponse.isValid()) {
|
||||
Log.w(TAG, String.format("tune_tryFrequency: saw response level %d at frequency %.2f", radioResponse.rssi, freqMHz));
|
||||
return radioResponse.rssi;
|
||||
} else {
|
||||
Log.w(TAG, "tune_tryFrequency: invalid radio response:" + ByteUtil.shortHexString(radioResponse.getPayload()));
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
public double quickTuneForPump(double startFrequencyMHz) {
|
||||
double betterFrequency = startFrequencyMHz;
|
||||
double stepsize = 0.05;
|
||||
for (int tries = 0; tries < 4; tries++) {
|
||||
double evenBetterFrequency = quickTunePumpStep(betterFrequency, stepsize);
|
||||
if (evenBetterFrequency == 0.0) {
|
||||
// could not see the pump at all.
|
||||
// Try again at larger step size
|
||||
stepsize += 0.05;
|
||||
} else {
|
||||
if ((int) (evenBetterFrequency * 100) == (int) (betterFrequency * 100)) {
|
||||
// value did not change, so we're done.
|
||||
break;
|
||||
}
|
||||
betterFrequency = evenBetterFrequency; // and go again.
|
||||
}
|
||||
}
|
||||
if (betterFrequency == 0.0) {
|
||||
// we've failed... caller should try a full scan for pump
|
||||
Log.e(TAG, "quickTuneForPump: failed to find pump");
|
||||
} else {
|
||||
rfspy.setBaseFrequency(betterFrequency);
|
||||
if (betterFrequency != startFrequencyMHz) {
|
||||
Log.i(TAG, String.format("quickTuneForPump: new frequency is %.2fMHz", betterFrequency));
|
||||
} else {
|
||||
Log.i(TAG, String.format("quickTuneForPump: pump frequency is the same: %.2fMHz", startFrequencyMHz));
|
||||
}
|
||||
}
|
||||
return betterFrequency;
|
||||
}
|
||||
|
||||
private double quickTunePumpStep(double startFrequencyMHz, double stepSizeMHz) {
|
||||
Log.i(TAG, "Doing quick radio tune for pump ID " + pumpID);
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
int startRssi = tune_tryFrequency(startFrequencyMHz);
|
||||
double lowerFrequency = startFrequencyMHz - stepSizeMHz;
|
||||
int lowerRssi = tune_tryFrequency(lowerFrequency);
|
||||
double higherFrequency = startFrequencyMHz + stepSizeMHz;
|
||||
int higherRssi = tune_tryFrequency(higherFrequency);
|
||||
if ((higherRssi == 0.0) && (lowerRssi == 0.0) && (startRssi == 0.0)) {
|
||||
// we can't see the pump at all...
|
||||
return 0.0;
|
||||
}
|
||||
if (higherRssi > startRssi) {
|
||||
// need to move higher
|
||||
return higherFrequency;
|
||||
} else if (lowerRssi > startRssi) {
|
||||
// need to move lower.
|
||||
return lowerFrequency;
|
||||
}
|
||||
return startFrequencyMHz;
|
||||
}
|
||||
|
||||
private double scanForPump(double[] frequencies) {
|
||||
Log.i(TAG, "Scanning for pump ID " + pumpID);
|
||||
wakeup(pumpAwakeForMinutes);
|
||||
FrequencyScanResults results = new FrequencyScanResults();
|
||||
|
||||
for (int i = 0; i < frequencies.length; i++) {
|
||||
int tries = 3;
|
||||
FrequencyTrial trial = new FrequencyTrial();
|
||||
trial.frequencyMHz = frequencies[i];
|
||||
rfspy.setBaseFrequency(frequencies[i]);
|
||||
int sumRSSI = 0;
|
||||
for (int j = 0; j < tries; j++) {
|
||||
PumpMessage msg = makePumpMessage(new MessageType(MessageType.GetPumpModel), new GetPumpModelCarelinkMessageBody());
|
||||
RFSpyResponse resp = rfspy.transmitThenReceive(new RadioPacket(msg.getTxData()), (byte) 0, (byte) 0, (byte) 0, (byte) 0, rfspy.EXPECTED_MAX_BLUETOOTH_LATENCY_MS, (byte) 0);
|
||||
if (resp.wasTimeout()) {
|
||||
Log.e(TAG, String.format("scanForPump: Failed to find pump at frequency %.2f", frequencies[i]));
|
||||
} else if (resp.looksLikeRadioPacket()) {
|
||||
RadioResponse radioResponse = new RadioResponse(resp.getRaw());
|
||||
if (radioResponse.isValid()) {
|
||||
sumRSSI += radioResponse.rssi;
|
||||
trial.successes++;
|
||||
} else {
|
||||
Log.w(TAG, "Failed to parse radio response: " + ByteUtil.shortHexString(resp.getRaw()));
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG, "scanForPump: raw response is " + ByteUtil.shortHexString(resp.getRaw()));
|
||||
}
|
||||
trial.tries++;
|
||||
}
|
||||
sumRSSI += -99.0 * (trial.tries - trial.successes);
|
||||
trial.averageRSSI = (double) (sumRSSI) / (double) (trial.tries);
|
||||
results.trials.add(trial);
|
||||
}
|
||||
results.sort(); // sorts in ascending order
|
||||
Log.d(TAG, "Sorted scan results:");
|
||||
for (int k = 0; k < results.trials.size(); k++) {
|
||||
FrequencyTrial one = results.trials.get(k);
|
||||
Log.d(TAG, String.format("Scan Result[%d]: Freq=%.2f, avg RSSI = %f", k, one.frequencyMHz, one.averageRSSI));
|
||||
}
|
||||
FrequencyTrial bestTrial = results.trials.get(results.trials.size() - 1);
|
||||
results.bestFrequencyMHz = bestTrial.frequencyMHz;
|
||||
if (bestTrial.successes > 0) {
|
||||
rfspy.setBaseFrequency(results.bestFrequencyMHz);
|
||||
return results.bestFrequencyMHz;
|
||||
} else {
|
||||
Log.e(TAG, "No pump response during scan.");
|
||||
return 0.0;
|
||||
}
|
||||
}
|
||||
|
||||
private PumpMessage makePumpMessage(MessageType messageType, MessageBody messageBody) {
|
||||
PumpMessage msg = new PumpMessage();
|
||||
msg.init(new PacketType(PacketType.Carelink), pumpID, messageType, messageBody);
|
||||
return msg;
|
||||
}
|
||||
|
||||
private PumpMessage makePumpMessage(byte msgType, MessageBody body) {
|
||||
return makePumpMessage(new MessageType(msgType), body);
|
||||
}
|
||||
|
||||
private PumpMessage makePumpMessage(byte[] typeAndBody) {
|
||||
PumpMessage msg = new PumpMessage();
|
||||
msg.init(ByteUtil.concat(ByteUtil.concat(new byte[]{(byte) 0xa7}, pumpID), typeAndBody));
|
||||
return msg;
|
||||
}
|
||||
|
||||
private void rememberLastGoodPumpCommunicationTime() {
|
||||
lastGoodPumpCommunicationTime = Instant.now();
|
||||
SharedPreferences.Editor ed = prefs.edit();
|
||||
ed.putLong("lastGoodReceiverCommunicationTime", lastGoodPumpCommunicationTime.getMillis());
|
||||
ed.commit();
|
||||
}
|
||||
|
||||
private Instant getLastGoodPumpCommunicationTime() {
|
||||
// If we have a value of zero, we need to load from prefs.
|
||||
if (lastGoodPumpCommunicationTime.getMillis() == new Instant(0).getMillis()) {
|
||||
lastGoodPumpCommunicationTime = new Instant(prefs.getLong("lastGoodReceiverCommunicationTime", 0));
|
||||
// Might still be zero, but that's fine.
|
||||
}
|
||||
double minutesAgo = (Instant.now().getMillis() - lastGoodPumpCommunicationTime.getMillis()) / (1000.0 * 60.0);
|
||||
Log.v(TAG, "Last good pump communication was " + minutesAgo + " minutes ago.");
|
||||
return lastGoodPumpCommunicationTime;
|
||||
}
|
||||
|
||||
public PumpMessage getRemainingBattery() {
|
||||
PumpMessage getBatteryMsg = makePumpMessage(new MessageType(MessageType.GetBattery), new CarelinkShortMessageBody());
|
||||
PumpMessage resp = sendAndListen(getBatteryMsg);
|
||||
return resp;
|
||||
}
|
||||
|
||||
public PumpMessage getRemainingInsulin() {
|
||||
PumpMessage msg = makePumpMessage(new MessageType(MessageType.CMD_M_READ_INSULIN_REMAINING), new CarelinkShortMessageBody());
|
||||
PumpMessage resp = sendAndListen(msg);
|
||||
return resp;
|
||||
}
|
||||
|
||||
public PumpMessage getCurrentBasalRate() {
|
||||
PumpMessage msg = makePumpMessage(new MessageType(MessageType.ReadTempBasal), new CarelinkShortMessageBody());
|
||||
PumpMessage resp = sendAndListen(msg);
|
||||
return resp;
|
||||
}
|
||||
|
||||
PumpManagerStatus pumpManagerStatus = new PumpManagerStatus();
|
||||
|
||||
public PumpManagerStatus getPumpManagerStatus() {
|
||||
return pumpManagerStatus;
|
||||
}
|
||||
|
||||
public void updatePumpManagerStatus() {
|
||||
PumpMessage resp = getRemainingBattery();
|
||||
if (resp.isValid()) {
|
||||
byte[] remainingBatteryBytes = resp.getContents();
|
||||
if (remainingBatteryBytes != null) {
|
||||
if (remainingBatteryBytes.length == 5) {
|
||||
/**
|
||||
* 0x72 0x03, 0x00, 0x00, 0x82
|
||||
* meaning what ????
|
||||
*/
|
||||
pumpManagerStatus.remainBattery = ByteUtil.asUINT8(remainingBatteryBytes[5]);
|
||||
}
|
||||
}
|
||||
}
|
||||
resp = getRemainingInsulin();
|
||||
byte[] insulinRemainingBytes = resp.getContents();
|
||||
if (insulinRemainingBytes != null) {
|
||||
if (insulinRemainingBytes.length == 4) {
|
||||
/* 0x73 0x02 0x05 0xd2
|
||||
* the 0xd2 (210) represents 21 units remaining.
|
||||
*/
|
||||
double insulinUnitsRemaining = ByteUtil.asUINT8(insulinRemainingBytes[3]) / 10.0;
|
||||
pumpManagerStatus.remainUnits = insulinUnitsRemaining;
|
||||
}
|
||||
}
|
||||
/* current basal */
|
||||
resp = getCurrentBasalRate();
|
||||
byte[] basalRateBytes = resp.getContents();
|
||||
if (basalRateBytes != null) {
|
||||
if (basalRateBytes.length == 2) {
|
||||
/**
|
||||
* 0x98 0x06
|
||||
* 0x98 is "basal rate"
|
||||
* 0x06 is what? Not currently running a temp basal, current basal is "standard" at 0
|
||||
*/
|
||||
double basalRate = ByteUtil.asUINT8(basalRateBytes[1]);
|
||||
pumpManagerStatus.currentBasal = basalRate;
|
||||
}
|
||||
}
|
||||
// get last bolus amount
|
||||
// get last bolus time
|
||||
// get tempBasalInProgress
|
||||
// get tempBasalRatio
|
||||
// get tempBasalRemainMin
|
||||
// get tempBasalStart
|
||||
// get pump time
|
||||
ReadPumpClockResult clockResult = getPumpRTC();
|
||||
if (clockResult.resultIsOK()) {
|
||||
pumpManagerStatus.time = clockResult.getTime().toDate();
|
||||
}
|
||||
// get last sync time
|
||||
|
||||
}
|
||||
|
||||
public void testPageDecode() {
|
||||
byte[] raw = new byte[]{(byte) 0x6D, (byte) 0x62, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x63, (byte) 0x10, (byte) 0x6D, (byte) 0x63, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x01,
|
||||
(byte) 0x01, (byte) 0x01, (byte) 0x00, (byte) 0x5A, (byte) 0xA5, (byte) 0x49, (byte) 0x04, (byte) 0x10, (byte) 0x01, (byte) 0x01, (byte) 0x01, (byte) 0x00, (byte) 0x6D, (byte) 0xA5, (byte) 0x49, (byte) 0x04, (byte) 0x10, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x08, (byte) 0x64, (byte) 0x10, (byte) 0x6D, (byte) 0x64, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x08, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x08, (byte) 0x64, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x08, (byte) 0x64, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x08, (byte) 0x64, (byte) 0x02, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x02, (byte) 0x0C, (byte) 0x00,
|
||||
(byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x64, (byte) 0x01, (byte) 0x75, (byte) 0x94, (byte) 0x0D, (byte) 0x05, (byte) 0x10, (byte) 0x64, (byte) 0x01, (byte) 0x44, (byte) 0x95, (byte) 0x0D, (byte) 0x05, (byte) 0x10, (byte) 0x17, (byte) 0x00, (byte) 0x4E, (byte) 0x95, (byte) 0x0D, (byte) 0x05, (byte) 0x10, (byte) 0x18, (byte) 0x00, (byte) 0x40, (byte) 0x95, (byte) 0x0D, (byte) 0x05, (byte) 0x10,
|
||||
(byte) 0x19, (byte) 0x00, (byte) 0x40, (byte) 0x81, (byte) 0x15, (byte) 0x05, (byte) 0x10, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x65, (byte) 0x10, (byte) 0x6D, (byte) 0x65, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x1A, (byte) 0x00, (byte) 0x47, (byte) 0x82, (byte) 0x09, (byte) 0x06,
|
||||
(byte) 0x10, (byte) 0x1A, (byte) 0x01, (byte) 0x5C, (byte) 0x82, (byte) 0x09, (byte) 0x06, (byte) 0x10, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x66, (byte) 0x10, (byte) 0x6D, (byte) 0x66, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x67, (byte) 0x10, (byte) 0x6D, (byte) 0x67, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x68, (byte) 0x10, (byte) 0x6D, (byte) 0x68, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x69, (byte) 0x10, (byte) 0x6D, (byte) 0x69, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x6A, (byte) 0x10, (byte) 0x6D, (byte) 0x6A, (byte) 0x10, (byte) 0x05, (byte) 0x0C,
|
||||
(byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x6B, (byte) 0x10, (byte) 0x6D, (byte) 0x6B, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x6C,
|
||||
(byte) 0x10, (byte) 0x6D, (byte) 0x6C, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x6D, (byte) 0x10, (byte) 0x6D, (byte) 0x6D, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x6E, (byte) 0x10, (byte) 0x6D, (byte) 0x6E, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x6F, (byte) 0x10, (byte) 0x6D, (byte) 0x6F, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00,
|
||||
(byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x19, (byte) 0x00, (byte) 0x40, (byte) 0x81, (byte) 0x03, (byte) 0x10, (byte) 0x10, (byte) 0x1A, (byte) 0x00, (byte) 0x68, (byte) 0x96, (byte) 0x0A, (byte) 0x10, (byte) 0x10, (byte) 0x1A, (byte) 0x01, (byte) 0x40, (byte) 0x97, (byte) 0x0A, (byte) 0x10, (byte) 0x10, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x70, (byte) 0x10, (byte) 0x6D, (byte) 0x70, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x71, (byte) 0x10, (byte) 0x6D, (byte) 0x71, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x72, (byte) 0x10, (byte) 0x6D, (byte) 0x72, (byte) 0x10, (byte) 0x05, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x73, (byte) 0x10, (byte) 0x6D, (byte) 0x73, (byte) 0x10, (byte) 0x05, (byte) 0x0C,
|
||||
(byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00,
|
||||
(byte) 0x00, (byte) 0x0C, (byte) 0x00, (byte) 0xE8, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x07, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x74, (byte) 0x10, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x2C, (byte) 0x79,
|
||||
};
|
||||
Page page = new Page();
|
||||
page.parseFrom(raw, PumpModel.MM522);
|
||||
page.parseByDates(raw, PumpModel.MM522);
|
||||
page.parsePicky(raw, PumpModel.MM522);
|
||||
Log.i(TAG, "testPageDecode: done");
|
||||
}
|
||||
|
||||
}
|
|
@ -1,61 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLink;
|
||||
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/16/16.
|
||||
*
|
||||
* This class is intended to provide compatibility with HAPP, and was
|
||||
* modeled after the PumpStatus class from happdanardriver
|
||||
*
|
||||
* happdanardriver/app/src/main/java/info/nightscout/danar/db/PumpStatus.java
|
||||
*/
|
||||
public class PumpManagerStatus {
|
||||
public long getTimeIndex() {
|
||||
return (long) Math.ceil(time.getTime() / 60000d );
|
||||
}
|
||||
|
||||
public void setTimeIndex(long timeIndex) {
|
||||
this.timeIndex = timeIndex;
|
||||
}
|
||||
|
||||
public long timeIndex;
|
||||
|
||||
public Date time;
|
||||
|
||||
public double remainUnits = 0;
|
||||
|
||||
public int remainBattery = 0;
|
||||
|
||||
public double currentBasal = 0;
|
||||
|
||||
public int tempBasalInProgress = 0;
|
||||
|
||||
public int tempBasalRatio = 0;
|
||||
|
||||
public int tempBasalRemainMin = 0 ;
|
||||
|
||||
public Date last_bolus_time ;
|
||||
|
||||
public double last_bolus_amount = 0;
|
||||
|
||||
public Date tempBasalStart;
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "PumpStatus{" +
|
||||
"timeIndex=" + timeIndex +
|
||||
", time=" + time +
|
||||
", remainUnits=" + remainUnits +
|
||||
", remainBattery=" + remainBattery +
|
||||
", currentBasal=" + currentBasal +
|
||||
", tempBasalInProgress=" + tempBasalInProgress +
|
||||
", tempBasalRatio=" + tempBasalRatio +
|
||||
", tempBasalRemainMin=" + tempBasalRemainMin +
|
||||
", last_bolus_time=" + last_bolus_time +
|
||||
", last_bolus_amount=" + last_bolus_amount +
|
||||
", tempBasalStart=" + tempBasalStart +
|
||||
'}';
|
||||
}
|
||||
|
||||
}
|
|
@ -1,45 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLink;
|
||||
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.CRC;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/4/16.
|
||||
*/
|
||||
public class RawHistoryPage {
|
||||
private static final String TAG = "RawHistoryPage";
|
||||
byte[] data = new byte[0];
|
||||
public RawHistoryPage() {
|
||||
}
|
||||
public void appendData(byte[] newdata) {
|
||||
data = ByteUtil.concat(data,newdata);
|
||||
}
|
||||
public byte[] getData() {
|
||||
return data;
|
||||
}
|
||||
public int getLength() {
|
||||
return data.length;
|
||||
}
|
||||
public boolean isChecksumOK() {
|
||||
if (getLength() != 1024) {
|
||||
return false;
|
||||
}
|
||||
byte[] computedCRC = CRC.calculate16CCITT(ByteUtil.substring(data,0,1022));
|
||||
return ((computedCRC[0] == data[1022]) && (computedCRC[1] == data[1023]));
|
||||
}
|
||||
|
||||
public void dumpToDebug() {
|
||||
int linesize = 80;
|
||||
int offset = 0;
|
||||
while (offset < data.length) {
|
||||
int bytesToLog = linesize;
|
||||
if (offset + linesize > data.length) {
|
||||
bytesToLog = data.length - offset;
|
||||
}
|
||||
Log.d(TAG,ByteUtil.shortHexString(ByteUtil.substring(data,offset,bytesToLog)));
|
||||
offset += linesize;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,20 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RileyLinkBLE;
|
||||
|
||||
import java.util.UUID;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public abstract class BLECommOperation {
|
||||
public boolean timedOut = false;
|
||||
public boolean interrupted = false;
|
||||
|
||||
// This is to be run on the main thread
|
||||
public abstract void execute(RileyLinkBLE comm);
|
||||
public void gattOperationCompletionCallback(UUID uuid, byte[] value) {}
|
||||
public int getGattOperationTimeout_ms() { return 22000;}
|
||||
|
||||
public byte[] getValue() { return null; }
|
||||
}
|
|
@ -1,16 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class BLECommOperationResult {
|
||||
public byte[] value;
|
||||
public int resultCode;
|
||||
|
||||
public static final int RESULT_NONE = 0;
|
||||
public static final int RESULT_SUCCESS = 1;
|
||||
public static final int RESULT_TIMEOUT = 2;
|
||||
public static final int RESULT_BUSY = 3;
|
||||
public static final int RESULT_INTERRUPTED = 4;
|
||||
public static final int RESULT_NOT_CONFIGURED = 5;
|
||||
}
|
|
@ -1,62 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations;
|
||||
|
||||
import android.bluetooth.BluetoothGatt;
|
||||
import android.bluetooth.BluetoothGattCharacteristic;
|
||||
import android.os.SystemClock;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.GattAttributes;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RileyLinkBLE;
|
||||
|
||||
import java.util.UUID;
|
||||
import java.util.concurrent.Semaphore;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class CharacteristicReadOperation extends BLECommOperation {
|
||||
private static final String TAG = "CharacteristicReadOp";
|
||||
private BluetoothGatt gatt;
|
||||
private BluetoothGattCharacteristic characteristic;
|
||||
private byte[] value;
|
||||
private Semaphore operationComplete = new Semaphore(0,true);
|
||||
public CharacteristicReadOperation(BluetoothGatt gatt, BluetoothGattCharacteristic chara) {
|
||||
this.gatt = gatt;
|
||||
this.characteristic = chara;
|
||||
}
|
||||
@Override
|
||||
public void execute(RileyLinkBLE comm) {
|
||||
gatt.readCharacteristic(characteristic);
|
||||
// wait here for callback to notify us that value was read.
|
||||
try {
|
||||
boolean didAcquire = operationComplete.tryAcquire(getGattOperationTimeout_ms(), TimeUnit.MILLISECONDS);
|
||||
if (didAcquire) {
|
||||
SystemClock.sleep(1); // This is to allow the IBinder thread to exit before we continue, allowing easier understanding of the sequence of events.
|
||||
// success
|
||||
} else {
|
||||
Log.e(TAG,"Timeout waiting for gatt write operation to complete");
|
||||
timedOut = true;
|
||||
}
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG,"Interrupted while waiting for gatt write operation to complete");
|
||||
interrupted = true;
|
||||
}
|
||||
value = characteristic.getValue();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void gattOperationCompletionCallback(UUID uuid, byte[] value) {
|
||||
super.gattOperationCompletionCallback(uuid, value);
|
||||
if (!characteristic.getUuid().equals(uuid)) {
|
||||
Log.e(TAG,String.format("Completion callback: UUID does not match! out of sequence? Found: %s, should be %s",
|
||||
GattAttributes.lookup(characteristic.getUuid()),GattAttributes.lookup(uuid)));
|
||||
}
|
||||
operationComplete.release();
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] getValue() {
|
||||
return value;
|
||||
}
|
||||
}
|
|
@ -1,66 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations;
|
||||
|
||||
import android.bluetooth.BluetoothGatt;
|
||||
import android.bluetooth.BluetoothGattCharacteristic;
|
||||
import android.os.SystemClock;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.GattAttributes;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RileyLinkBLE;
|
||||
|
||||
import java.util.UUID;
|
||||
import java.util.concurrent.Semaphore;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class CharacteristicWriteOperation extends BLECommOperation {
|
||||
private static final String TAG = "CharacteristicWriteOp";
|
||||
private BluetoothGatt gatt;
|
||||
private BluetoothGattCharacteristic characteristic;
|
||||
private byte[] value;
|
||||
private Semaphore operationComplete = new Semaphore(0,true);
|
||||
|
||||
public CharacteristicWriteOperation(BluetoothGatt gatt, BluetoothGattCharacteristic chara, byte[] value) {
|
||||
this.gatt = gatt;
|
||||
this.characteristic = chara;
|
||||
this.value = value;
|
||||
}
|
||||
@Override
|
||||
public void execute(RileyLinkBLE comm) {
|
||||
|
||||
characteristic.setValue(value);
|
||||
gatt.writeCharacteristic(characteristic);
|
||||
// wait here for callback to notify us that value was written.
|
||||
try {
|
||||
boolean didAcquire = operationComplete.tryAcquire(getGattOperationTimeout_ms(), TimeUnit.MILLISECONDS);
|
||||
if (didAcquire) {
|
||||
SystemClock.sleep(1); // This is to allow the IBinder thread to exit before we continue, allowing easier understanding of the sequence of events.
|
||||
// success
|
||||
} else {
|
||||
Log.e(TAG,"Timeout waiting for gatt write operation to complete");
|
||||
timedOut = true;
|
||||
}
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG,"Interrupted while waiting for gatt write operation to complete");
|
||||
interrupted = true;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// This will be run on the IBinder thread
|
||||
@Override
|
||||
public void gattOperationCompletionCallback(UUID uuid, byte[] value) {
|
||||
if (!characteristic.getUuid().equals(uuid)) {
|
||||
Log.e(TAG,String.format("Completion callback: UUID does not match! out of sequence? Found: %s, should be %s",
|
||||
GattAttributes.lookup(characteristic.getUuid()),GattAttributes.lookup(uuid)));
|
||||
}
|
||||
operationComplete.release();
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] getValue() {
|
||||
return value;
|
||||
}
|
||||
}
|
|
@ -1,56 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations;
|
||||
|
||||
import android.bluetooth.BluetoothGatt;
|
||||
import android.bluetooth.BluetoothGattDescriptor;
|
||||
import android.os.SystemClock;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RileyLinkBLE;
|
||||
|
||||
import java.util.UUID;
|
||||
import java.util.concurrent.Semaphore;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class DescriptorWriteOperation extends BLECommOperation {
|
||||
|
||||
private static final String TAG = "DescrWriteOp";
|
||||
private BluetoothGatt gatt;
|
||||
private BluetoothGattDescriptor descr;
|
||||
private byte[] value;
|
||||
private Semaphore operationComplete = new Semaphore(0,true);
|
||||
|
||||
public DescriptorWriteOperation(BluetoothGatt gatt, BluetoothGattDescriptor descr, byte[] value) {
|
||||
this.gatt = gatt;
|
||||
this.descr = descr;
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void gattOperationCompletionCallback(UUID uuid, byte[] value) {
|
||||
super.gattOperationCompletionCallback(uuid, value);
|
||||
operationComplete.release();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void execute(RileyLinkBLE comm) {
|
||||
descr.setValue(value);
|
||||
gatt.writeDescriptor(descr);
|
||||
// wait here for callback to notify us that value was read.
|
||||
try {
|
||||
boolean didAcquire = operationComplete.tryAcquire(getGattOperationTimeout_ms(), TimeUnit.MILLISECONDS);
|
||||
if (didAcquire) {
|
||||
SystemClock.sleep(1); // This is to allow the IBinder thread to exit before we continue, allowing easier understanding of the sequence of events.
|
||||
// success
|
||||
} else {
|
||||
Log.e(TAG,"Timeout waiting for descriptor write operation to complete");
|
||||
timedOut = true;
|
||||
}
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG,"Interrupted while waiting for descriptor write operation to complete");
|
||||
interrupted = true;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,52 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
import java.util.HashMap;
|
||||
import java.util.UUID;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/21/16.
|
||||
*/
|
||||
public class GattAttributes {
|
||||
// NOTE: these uuid strings must be lower case!
|
||||
private static HashMap<String, String> attributes = new HashMap();
|
||||
public static String PREFIX = "0000";
|
||||
public static String SUFFIX = "-0000-1000-8000-00805f9b34fb";
|
||||
public static String SERVICE_GAP = PREFIX + "1800" + SUFFIX;
|
||||
public static String CHARA_GAP_NAME = PREFIX + "2a00" + SUFFIX;
|
||||
public static String CHARA_GAP_NUM = PREFIX + "2a01" + SUFFIX;
|
||||
|
||||
public static String SERVICE_BATTERY = PREFIX + "180f" + SUFFIX;
|
||||
public static String CHARA_BATTERY_UNK = PREFIX + "2a19" + SUFFIX;
|
||||
|
||||
public static String SERVICE_RADIO = "0235733b-99c5-4197-b856-69219c2a3845";
|
||||
public static String CHARA_RADIO_DATA = "c842e849-5028-42e2-867c-016adada9155";
|
||||
public static String CHARA_RADIO_RESPONSE_COUNT = "6e6c7910-b89e-43a5-a0fe-50c5e2b81f4a";
|
||||
public static String CHARA_RADIO_TIMER_TICK = "6e6c7910-b89e-43a5-78af-50c5e2b86f7e";
|
||||
public static String CHARA_RADIO_CUSTOM_NAME = "d93b2af0-1e28-11e4-8c21-0800200c9a66";
|
||||
public static String CHARA_RADIO_VERSION = "30d99dc9-7c91-4295-a051-0a104d238cf2";
|
||||
|
||||
// table of names for uuids
|
||||
static {
|
||||
attributes.put(SERVICE_GAP, "Device Information Service");
|
||||
attributes.put(SERVICE_BATTERY,"Battery Service");
|
||||
attributes.put(CHARA_RADIO_CUSTOM_NAME,"customName");
|
||||
attributes.put(CHARA_RADIO_DATA,"data");
|
||||
attributes.put(CHARA_RADIO_RESPONSE_COUNT,"responseCount");
|
||||
attributes.put(CHARA_RADIO_TIMER_TICK,"timerTick");
|
||||
attributes.put(CHARA_RADIO_VERSION,"radioVersion");
|
||||
}
|
||||
|
||||
public static String lookup(UUID uuid) {
|
||||
return lookup(uuid.toString());
|
||||
}
|
||||
|
||||
public static String lookup(String uuid) {
|
||||
return lookup(uuid, uuid);
|
||||
}
|
||||
|
||||
public static String lookup(String uuid, String defaultName) {
|
||||
String name = attributes.get(uuid);
|
||||
return name == null ? defaultName : name;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,203 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
import android.content.Context;
|
||||
import android.os.SystemClock;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.BLECommOperation;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.BLECommOperationResult;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.StringUtil;
|
||||
import com.gxwtech.roundtrip2.util.ThreadUtil;
|
||||
|
||||
import java.util.UUID;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class RFSpy {
|
||||
public static final byte RFSPY_GET_STATE = 1;
|
||||
public static final byte RFSPY_GET_VERSION = 2;
|
||||
public static final byte RFSPY_GET_PACKET = 3; // aka Listen, receive
|
||||
public static final byte RFSPY_SEND = 4;
|
||||
public static final byte RFSPY_SEND_AND_LISTEN = 5;
|
||||
public static final byte RFSPY_UPDATE_REGISTER = 6;
|
||||
public static final byte RFSPY_RESET = 7;
|
||||
|
||||
public static final long RILEYLINK_FREQ_XTAL = 24000000;
|
||||
|
||||
public static final byte CC111X_REG_FREQ2 = 0x09;
|
||||
public static final byte CC111X_REG_FREQ1 = 0x0A;
|
||||
public static final byte CC111X_REG_FREQ0 = 0x0B;
|
||||
public static final byte CC111X_MDMCFG4 = 0x0C;
|
||||
public static final byte CC111X_MDMCFG3 = 0x0D;
|
||||
public static final byte CC111X_MDMCFG2 = 0x0E;
|
||||
public static final byte CC111X_MDMCFG1 = 0x0F;
|
||||
public static final byte CC111X_MDMCFG0 = 0x10;
|
||||
public static final byte CC111X_AGCCTRL2 = 0x17;
|
||||
public static final byte CC111X_AGCCTRL1 = 0x18;
|
||||
public static final byte CC111X_AGCCTRL0 = 0x19;
|
||||
public static final byte CC111X_FREND1 = 0x1A;
|
||||
public static final byte CC111X_FREND0 = 0x1B;
|
||||
|
||||
public static final int EXPECTED_MAX_BLUETOOTH_LATENCY_MS = 1500;
|
||||
|
||||
private static final String TAG = "RFSpy";
|
||||
private RileyLinkBLE rileyLinkBle;
|
||||
private RFSpyReader reader;
|
||||
private Context context;
|
||||
UUID radioServiceUUID = UUID.fromString(GattAttributes.SERVICE_RADIO);
|
||||
UUID radioDataUUID = UUID.fromString(GattAttributes.CHARA_RADIO_DATA);
|
||||
UUID radioVersionUUID = UUID.fromString(GattAttributes.CHARA_RADIO_VERSION);
|
||||
UUID responseCountUUID = UUID.fromString(GattAttributes.CHARA_RADIO_RESPONSE_COUNT);
|
||||
|
||||
private static final byte[] pumpID = {0x51, (byte)0x81, 0x63};
|
||||
|
||||
public RFSpy(Context context, RileyLinkBLE rileyLinkBle) {
|
||||
this.context = context;
|
||||
this.rileyLinkBle = rileyLinkBle;
|
||||
reader = new RFSpyReader(context, rileyLinkBle);
|
||||
}
|
||||
|
||||
// Call this after the RL services are discovered.
|
||||
// Starts an async task to read when data is available
|
||||
public void startReader() {
|
||||
rileyLinkBle.registerRadioResponseCountNotification(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
newDataIsAvailable();
|
||||
}
|
||||
});
|
||||
reader.start();
|
||||
}
|
||||
|
||||
// Call this from the "response count" notification handler.
|
||||
public void newDataIsAvailable() {
|
||||
// pass the message to the reader (which should be internal to RFSpy)
|
||||
reader.newDataIsAvailable();
|
||||
}
|
||||
|
||||
// This gets the version from the BLE113, not from the CC1110.
|
||||
// I.e., this gets the version from the BLE interface, not from the radio.
|
||||
public String getVersion() {
|
||||
BLECommOperationResult result = rileyLinkBle.readCharacteristic_blocking(radioServiceUUID,radioVersionUUID);
|
||||
if (result.resultCode == BLECommOperationResult.RESULT_SUCCESS) {
|
||||
return StringUtil.fromBytes(result.value);
|
||||
} else {
|
||||
Log.e(TAG,"getVersion failed with code: "+result.resultCode);
|
||||
return "(null)";
|
||||
}
|
||||
}
|
||||
|
||||
// The caller has to know how long the RFSpy will be busy with what was sent to it.
|
||||
private RFSpyResponse writeToData(byte[] bytes, int responseTimeout_ms) {
|
||||
SystemClock.sleep(100);
|
||||
// FIXME drain read queue?
|
||||
byte[] junkInBuffer = reader.poll(0);
|
||||
|
||||
while (junkInBuffer != null) {
|
||||
Log.w(TAG, ThreadUtil.sig()+"writeToData: draining read queue, found this: " + ByteUtil.shortHexString(junkInBuffer));
|
||||
junkInBuffer = reader.poll(0);
|
||||
}
|
||||
|
||||
// prepend length, and send it.
|
||||
byte[] prepended = ByteUtil.concat(new byte[] {(byte)(bytes.length)},bytes);
|
||||
BLECommOperationResult writeCheck = rileyLinkBle.writeCharacteristic_blocking(radioServiceUUID,radioDataUUID,prepended);
|
||||
if (writeCheck.resultCode != BLECommOperationResult.RESULT_SUCCESS) {
|
||||
Log.e(TAG,"BLE Write operation failed, code=" + writeCheck.resultCode);
|
||||
return new RFSpyResponse(); // will be a null (invalid) response
|
||||
}
|
||||
SystemClock.sleep(100);
|
||||
//Log.i(TAG,ThreadUtil.sig()+String.format(" writeToData:(timeout %d) %s",(responseTimeout_ms),ByteUtil.shortHexString(prepended)));
|
||||
byte[] rawResponse = reader.poll(responseTimeout_ms);
|
||||
RFSpyResponse resp = new RFSpyResponse(rawResponse);
|
||||
if (rawResponse == null) {
|
||||
Log.e(TAG,"writeToData: No response from RileyLink");
|
||||
} else {
|
||||
if (resp.wasInterrupted()) {
|
||||
Log.e(TAG, "writeToData: RileyLink was interrupted");
|
||||
} else if (resp.wasTimeout()) {
|
||||
Log.e(TAG, "writeToData: RileyLink reports timeout");
|
||||
} else if (resp.isOK()) {
|
||||
Log.w(TAG, "writeToData: RileyLink reports OK");
|
||||
} else {
|
||||
if (resp.looksLikeRadioPacket()) {
|
||||
RadioResponse radioResp = resp.getRadioResponse();
|
||||
byte[] responsePayload = radioResp.getPayload();
|
||||
Log.i(TAG,"writeToData: decoded radio response is "+ByteUtil.shortHexString(responsePayload));
|
||||
}
|
||||
//Log.i(TAG, "writeToData: raw response is " + ByteUtil.shortHexString(rawResponse));
|
||||
}
|
||||
}
|
||||
return resp;
|
||||
}
|
||||
|
||||
public RFSpyResponse getRadioVersion() {
|
||||
RFSpyResponse resp = writeToData(new byte[] {RFSPY_GET_VERSION},1000);
|
||||
if (resp == null) {
|
||||
Log.e(TAG,"getRadioVersion returned null");
|
||||
}
|
||||
/*
|
||||
Log.d(TAG,"checking response count");
|
||||
BLECommOperationResult checkRC = rileyLinkBle.readCharacteristic_blocking(radioServiceUUID,responseCountUUID);
|
||||
if (checkRC.resultCode == BLECommOperationResult.RESULT_SUCCESS) {
|
||||
Log.d(TAG,"Response count is: " + ByteUtil.shortHexString(checkRC.value));
|
||||
} else {
|
||||
Log.e(TAG,"Error getting response count, code is " + checkRC.resultCode);
|
||||
}
|
||||
*/
|
||||
return resp;
|
||||
}
|
||||
|
||||
public RFSpyResponse transmit(RadioPacket radioPacket, byte sendChannel, byte repeatCount, byte delay_ms) {
|
||||
// append checksum, encode data, send it.
|
||||
byte[] fullPacket = ByteUtil.concat(new byte[] {RFSPY_SEND,sendChannel,repeatCount, delay_ms},radioPacket.getEncoded());
|
||||
RFSpyResponse response = writeToData(fullPacket,repeatCount * delay_ms);
|
||||
return response;
|
||||
}
|
||||
|
||||
public RFSpyResponse receive(byte listenChannel, int timeout_ms, byte retryCount) {
|
||||
int receiveDelay = timeout_ms * (retryCount+1);
|
||||
byte[] listen = {RFSPY_GET_PACKET,listenChannel,
|
||||
(byte)((timeout_ms >> 24)&0x0FF),
|
||||
(byte)((timeout_ms >> 16)&0x0FF),
|
||||
(byte)((timeout_ms >> 8)&0x0FF),
|
||||
(byte)(timeout_ms & 0x0FF),
|
||||
retryCount};
|
||||
return writeToData(listen,receiveDelay);
|
||||
}
|
||||
|
||||
public RFSpyResponse transmitThenReceive(RadioPacket pkt, int timeout_ms) {
|
||||
return transmitThenReceive(pkt,(byte)0,(byte)0,(byte)0,(byte)0,timeout_ms,(byte)0);
|
||||
}
|
||||
|
||||
public RFSpyResponse transmitThenReceive(RadioPacket pkt, byte sendChannel, byte repeatCount, byte delay_ms, byte listenChannel, int timeout_ms, byte retryCount) {
|
||||
|
||||
int sendDelay = repeatCount * delay_ms;
|
||||
int receiveDelay = timeout_ms * (retryCount + 1);
|
||||
byte[] sendAndListen = {RFSPY_SEND_AND_LISTEN,sendChannel,repeatCount,delay_ms,listenChannel,
|
||||
(byte)((timeout_ms >> 24)&0x0FF),
|
||||
(byte)((timeout_ms >> 16)&0x0FF),
|
||||
(byte)((timeout_ms >> 8)&0x0FF),
|
||||
(byte)(timeout_ms & 0x0FF),
|
||||
retryCount};
|
||||
byte[] fullPacket = ByteUtil.concat(sendAndListen,pkt.getEncoded());
|
||||
return writeToData(fullPacket, sendDelay + receiveDelay + EXPECTED_MAX_BLUETOOTH_LATENCY_MS);
|
||||
}
|
||||
|
||||
public RFSpyResponse updateRegister(byte addr, byte val) {
|
||||
byte[] updateRegisterPkt = new byte[] {RFSPY_UPDATE_REGISTER,addr,val};
|
||||
RFSpyResponse resp = writeToData(updateRegisterPkt,EXPECTED_MAX_BLUETOOTH_LATENCY_MS);
|
||||
return resp;
|
||||
}
|
||||
|
||||
public void setBaseFrequency(double freqMHz) {
|
||||
int value = (int)(freqMHz * 1000000/((double)(RILEYLINK_FREQ_XTAL)/Math.pow(2.0,16.0)));
|
||||
updateRegister(CC111X_REG_FREQ0, (byte)(value & 0xff));
|
||||
updateRegister(CC111X_REG_FREQ1, (byte)((value >> 8) & 0xff));
|
||||
updateRegister(CC111X_REG_FREQ2, (byte)((value >> 16) & 0xff));
|
||||
Log.w(TAG,String.format("Set frequency to %.2f",freqMHz));
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,117 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
import android.content.Context;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.SystemClock;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.BLECommOperationResult;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.ThreadUtil;
|
||||
|
||||
import java.util.UUID;
|
||||
import java.util.concurrent.LinkedBlockingQueue;
|
||||
import java.util.concurrent.Semaphore;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class RFSpyReader {
|
||||
private static final String TAG = "RFSpyReader";
|
||||
private Context context;
|
||||
private RileyLinkBLE rileyLinkBle;
|
||||
private Semaphore waitForRadioData = new Semaphore(0,true);
|
||||
AsyncTask<Void,Void,Void> readerTask;
|
||||
private LinkedBlockingQueue<byte[]> mDataQueue = new LinkedBlockingQueue<>();
|
||||
private int acquireCount = 0;
|
||||
private int releaseCount = 0;
|
||||
|
||||
public RFSpyReader(Context context, RileyLinkBLE rileyLinkBle) {
|
||||
this.context = context;
|
||||
this.rileyLinkBle = rileyLinkBle;
|
||||
}
|
||||
|
||||
public void init(Context context, RileyLinkBLE rileyLinkBLE) {
|
||||
this.context = context;
|
||||
this.rileyLinkBle = rileyLinkBLE;
|
||||
}
|
||||
|
||||
public void setRileyLinkBle(RileyLinkBLE rileyLinkBle) {
|
||||
if (readerTask != null) {
|
||||
readerTask.cancel(true);
|
||||
}
|
||||
this.rileyLinkBle = rileyLinkBle;
|
||||
}
|
||||
|
||||
// This timeout must be coordinated with the length of the RFSpy radio operation or Bad Things Happen.
|
||||
public byte[] poll(int timeout_ms) {
|
||||
Log.v(TAG, ThreadUtil.sig()+"Entering poll at t=="+SystemClock.uptimeMillis()+", timeout is "+timeout_ms+" mDataQueue size is "+mDataQueue.size());
|
||||
if (mDataQueue.isEmpty())
|
||||
try {
|
||||
// block until timeout or data available.
|
||||
// returns null if timeout.
|
||||
byte[] dataFromQueue = mDataQueue.poll(timeout_ms, TimeUnit.MILLISECONDS);
|
||||
if (dataFromQueue != null) {
|
||||
Log.d(TAG, "Got data [" + ByteUtil.shortHexString(dataFromQueue) + "] at t==" + SystemClock.uptimeMillis());
|
||||
} else {
|
||||
Log.d(TAG, "Got data [null] at t==" + SystemClock.uptimeMillis());
|
||||
}
|
||||
return dataFromQueue;
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG,"poll: Interrupted waiting for data");
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
// Call this from the "response count" notification handler.
|
||||
public void newDataIsAvailable() {
|
||||
releaseCount++;
|
||||
Log.v(TAG,ThreadUtil.sig()+"waitForRadioData released(count="+releaseCount+") at t="+SystemClock.uptimeMillis());
|
||||
waitForRadioData.release();
|
||||
}
|
||||
|
||||
public void start() {
|
||||
readerTask = new AsyncTask<Void, Void, Void>() {
|
||||
@Override
|
||||
protected Void doInBackground(Void... voids) {
|
||||
UUID serviceUUID = UUID.fromString(GattAttributes.SERVICE_RADIO);
|
||||
UUID radioDataUUID = UUID.fromString(GattAttributes.CHARA_RADIO_DATA);
|
||||
BLECommOperationResult result;
|
||||
while (true) {
|
||||
try {
|
||||
acquireCount++;
|
||||
waitForRadioData.acquire();
|
||||
Log.v(TAG,ThreadUtil.sig()+"waitForRadioData acquired (count="+acquireCount+") at t="+SystemClock.uptimeMillis());
|
||||
SystemClock.sleep(100);
|
||||
SystemClock.sleep(1);
|
||||
result = rileyLinkBle.readCharacteristic_blocking(serviceUUID, radioDataUUID);
|
||||
SystemClock.sleep(100);
|
||||
|
||||
if (result.resultCode == BLECommOperationResult.RESULT_SUCCESS) {
|
||||
// only data up to the first null is valid
|
||||
for (int i=0; i<result.value.length; i++) {
|
||||
if (result.value[i]==0) {
|
||||
result.value = ByteUtil.substring(result.value,0,i);
|
||||
break;
|
||||
}
|
||||
}
|
||||
mDataQueue.add(result.value);
|
||||
} else if (result.resultCode == BLECommOperationResult.RESULT_INTERRUPTED) {
|
||||
Log.e(TAG, "Read operation was interrupted");
|
||||
} else if (result.resultCode == BLECommOperationResult.RESULT_TIMEOUT) {
|
||||
Log.e(TAG, "Read operation on Radio Data timed out");
|
||||
} else if (result.resultCode == BLECommOperationResult.RESULT_BUSY) {
|
||||
Log.e(TAG, "FAIL: RileyLinkBLE reports operation already in progress");
|
||||
} else if (result.resultCode == BLECommOperationResult.RESULT_NONE) {
|
||||
Log.e(TAG, "FAIL: got invalid result code: " + result.resultCode);
|
||||
}
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG, "Interrupted while waiting for data");
|
||||
}
|
||||
}
|
||||
}
|
||||
}.execute();
|
||||
}
|
||||
|
||||
}
|
|
@ -1,75 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class RFSpyResponse {
|
||||
// 0xaa == timeout
|
||||
// 0xbb == interrupted
|
||||
// 0xcc == zero-data
|
||||
protected byte[] raw;
|
||||
protected RadioResponse radioResponse;
|
||||
|
||||
public RFSpyResponse() {
|
||||
init(new byte[0]);
|
||||
}
|
||||
|
||||
public RFSpyResponse(byte[] bytes) {
|
||||
init(bytes);
|
||||
}
|
||||
|
||||
public void init(byte[] bytes) {
|
||||
if (bytes == null) {
|
||||
raw = new byte[0];
|
||||
} else {
|
||||
raw = bytes;
|
||||
}
|
||||
if (looksLikeRadioPacket()) {
|
||||
radioResponse = new RadioResponse(raw);
|
||||
} else {
|
||||
radioResponse = new RadioResponse();
|
||||
}
|
||||
}
|
||||
|
||||
public RadioResponse getRadioResponse() {
|
||||
return radioResponse;
|
||||
}
|
||||
|
||||
public boolean wasTimeout() {
|
||||
if ((raw.length == 1)||(raw.length==2)) {
|
||||
if (raw[0] == (byte)0xaa) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public boolean wasInterrupted() {
|
||||
if ((raw.length == 1)||(raw.length==2)) {
|
||||
if (raw[0] == (byte)0xbb) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public boolean isOK() {
|
||||
if ((raw.length == 1)||(raw.length==2)) {
|
||||
if (raw[0] == (byte)0x01) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public boolean looksLikeRadioPacket() {
|
||||
if (raw.length > 2) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public byte[] getRaw() {
|
||||
return raw;
|
||||
}
|
||||
}
|
|
@ -1,295 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.CRC;
|
||||
|
||||
import java.util.ArrayList;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/31/15.
|
||||
*/
|
||||
public class RFTools {
|
||||
private static final String TAG = "RFTools";
|
||||
/*
|
||||
CodeSymbols is an ordered list of translations
|
||||
6bits -> 4 bits, in order from 0x0 to 0xF
|
||||
The 6 bit codes are what is used on the RF side of the RileyLink
|
||||
to communicate with a Medtronic pump.
|
||||
|
||||
*/
|
||||
public static byte[] CodeSymbols = {
|
||||
0x15,
|
||||
0x31,
|
||||
0x32,
|
||||
0x23,
|
||||
0x34,
|
||||
0x25,
|
||||
0x26,
|
||||
0x16,
|
||||
0x1a,
|
||||
0x19,
|
||||
0x2a,
|
||||
0x0b,
|
||||
0x2c,
|
||||
0x0d,
|
||||
0x0e,
|
||||
0x1c
|
||||
};
|
||||
|
||||
public static byte[] appendChecksum(final byte[] input) {
|
||||
if (input == null) {
|
||||
return null;
|
||||
}
|
||||
if (input.length == 0) {
|
||||
return null;
|
||||
}
|
||||
byte[] rval = new byte[input.length+1];
|
||||
System.arraycopy(input, 0, rval, 0, input.length);
|
||||
byte mycrc = CRC.crc8(input);
|
||||
Log.d(TAG,String.format("Adding checksum 0x%02X to %d byte array from 0x%02X to 0x%02X",mycrc,input.length,input[0],input[input.length-1]));
|
||||
rval[input.length] = mycrc;
|
||||
return rval;
|
||||
}
|
||||
|
||||
public static ArrayList<Byte> fromBytes(byte[] data) {
|
||||
ArrayList<Byte> rval = new ArrayList<>();
|
||||
for (int i=0; i<data.length; i++) {
|
||||
rval.add(data[i]);
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
public static byte[] toBytes(ArrayList<Byte> data) {
|
||||
byte[] rval = new byte[data.size()];
|
||||
for (int i = 0; i < data.size(); i++) {
|
||||
rval[i] = data.get(i);
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
/*
|
||||
+ (NSData*)encode4b6b:(NSData*)data {
|
||||
NSMutableData *outData = [NSMutableData data];
|
||||
NSMutableData *dataPlusCrc = [data mutableCopy];
|
||||
unsigned char crc = [MinimedPacket crcForData:data];
|
||||
[dataPlusCrc appendBytes:&crc length:1];
|
||||
char codes[16] = {21,49,50,35,52,37,38,22,26,25,42,11,44,13,14,28};
|
||||
const unsigned char *inBytes = [dataPlusCrc bytes];
|
||||
unsigned int acc = 0x0;
|
||||
int bitcount = 0;
|
||||
for (int i=0; i < dataPlusCrc.length; i++) {
|
||||
acc <<= 6;
|
||||
acc |= codes[inBytes[i] >> 4];
|
||||
bitcount += 6;
|
||||
|
||||
acc <<= 6;
|
||||
acc |= codes[inBytes[i] & 0x0f];
|
||||
bitcount += 6;
|
||||
|
||||
while (bitcount >= 8) {
|
||||
unsigned char outByte = acc >> (bitcount-8) & 0xff;
|
||||
[outData appendBytes:&outByte length:1];
|
||||
bitcount -= 8;
|
||||
acc &= (0xffff >> (16-bitcount));
|
||||
}
|
||||
}
|
||||
if (bitcount > 0) {
|
||||
acc <<= (8-bitcount);
|
||||
unsigned char outByte = acc & 0xff;
|
||||
[outData appendBytes:&outByte length:1];
|
||||
}
|
||||
return outData;
|
||||
}
|
||||
*/
|
||||
|
||||
public static final byte[] codes = new byte[] {21,49,50,35,52,37,38,22,26,25,42,11,44,13,14,28 };
|
||||
|
||||
/* O(n) lookup. Run on an O(n) translation of a byte-stream, gives O(n**2) performance. Sigh. */
|
||||
public static int codeIndex(byte b) {
|
||||
for (int i=0; i< codes.length; i++) {
|
||||
if (b == codes[i]) {
|
||||
return i;
|
||||
}
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
public static byte[] encode4b6b(byte[] data) {
|
||||
if ((data.length % 2)!=0) {
|
||||
// Log.e(TAG,"Warning: data is odd number of bytes");
|
||||
}
|
||||
// use arraylists because byte[] is annoying.
|
||||
ArrayList<Byte> inData = fromBytes(data);
|
||||
ArrayList<Byte> outData = new ArrayList<>();
|
||||
|
||||
int acc = 0;
|
||||
int bitcount = 0;
|
||||
int i;
|
||||
for (i=0; i<inData.size(); i++) {
|
||||
acc <<= 6;
|
||||
acc |= codes[(inData.get(i) >> 4) & 0x0f];
|
||||
bitcount += 6;
|
||||
|
||||
acc <<= 6;
|
||||
acc |= codes[inData.get(i) & 0x0f];
|
||||
bitcount += 6;
|
||||
|
||||
while (bitcount >= 8) {
|
||||
byte outByte = (byte)(acc >> (bitcount-8) & 0xff);
|
||||
outData.add(outByte);
|
||||
bitcount -= 8;
|
||||
acc &= (0xffff >> (16 - bitcount));
|
||||
}
|
||||
}
|
||||
if (bitcount > 0) {
|
||||
acc <<= 6;
|
||||
acc |= 0x14; // marks uneven packet boundary.
|
||||
bitcount += 6;
|
||||
if (bitcount >= 8) {
|
||||
byte outByte = (byte)((acc >> (bitcount - 8)) & 0xff);
|
||||
outData.add(outByte);
|
||||
bitcount -=8;
|
||||
// acc &= (0xffff >> (16 - bitcount));
|
||||
}
|
||||
while(bitcount >=8) {
|
||||
outData.add((byte)0);
|
||||
bitcount -= 8;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// convert back to byte[]
|
||||
byte[] rval = toBytes(outData);
|
||||
|
||||
return rval;
|
||||
|
||||
}
|
||||
|
||||
public static void test() {
|
||||
/*
|
||||
{0xa7} -> {0xa9, 0x60}
|
||||
{0xa7, 0x12} -> {0xa9, 0x6c, 0x72}
|
||||
{0xa7, 0x12, 0xa7} -> {0xa9, 0x6c, 0x72, 0xa9, 0x60}
|
||||
*/
|
||||
/* test compare */
|
||||
byte[] s1 = {0,1,2};
|
||||
byte[] s2 = {2,1,0,3};
|
||||
byte[] s3 = {0,1,2,3};
|
||||
if (ByteUtil.compare(s1,s1)!=0) {
|
||||
Log.e(TAG,"test: compare failed.");
|
||||
}
|
||||
if (ByteUtil.compare(s1,s2)>=0) {
|
||||
Log.e(TAG,"test: compare failed.");
|
||||
}
|
||||
if (ByteUtil.compare(s2,s1)<=0) {
|
||||
Log.e(TAG,"test: compare failed.");
|
||||
}
|
||||
if (ByteUtil.compare(s1,s3)>=0) {
|
||||
Log.e(TAG,"test: compare failed.");
|
||||
}
|
||||
//testCompose(new byte[] {(byte)0xa7, (byte)0xa7});
|
||||
byte[] bs = encode4b6b(new byte[]{(byte) 0xa7});
|
||||
byte[] out = new byte[] {(byte)(0xa9),0x65};
|
||||
if (ByteUtil.compare(bs,out)!=0) {
|
||||
Log.e(TAG,"encode Data failed: expected "+ByteUtil.shortHexString(out)+" but got "+ByteUtil.shortHexString(bs));
|
||||
}
|
||||
bs = encode4b6b(new byte[]{(byte) 0xa7, 0x12});
|
||||
out = new byte[] {(byte)(0xa9),0x6c,0x72};
|
||||
if (ByteUtil.compare(bs,out)!=0) {
|
||||
Log.e(TAG,"encode Data failed: expected "+ByteUtil.shortHexString(out)+" but got "+ByteUtil.shortHexString(bs));
|
||||
}
|
||||
bs = encode4b6b(new byte[]{(byte) 0xa7, 0x12, (byte) 0xa7});
|
||||
out = new byte[] {(byte)(0xa9),0x6c,0x72,(byte)0xa9,0x65};
|
||||
if (ByteUtil.compare(bs,out)!=0) {
|
||||
Log.e(TAG,"encode Data failed: expected "+ByteUtil.shortHexString(out)+" but got "+ByteUtil.shortHexString(bs));
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
public static byte[] decode4b6b(byte[] raw) throws NumberFormatException {
|
||||
/*
|
||||
if ((raw.length % 2) != 0) {
|
||||
Log.e(TAG,"Warning: data is odd number of bytes");
|
||||
}
|
||||
*/
|
||||
byte[] rval = new byte[]{};
|
||||
int availableBits = 0;
|
||||
int codingErrors = 0;
|
||||
int x = 0;
|
||||
//Log.w(TAG,"decode4b6b: untested code");
|
||||
//Log.w(TAG,String.format("Decoding %d bytes: %s",raw.length,ByteUtil.shortHexString(raw)));
|
||||
for (int i=0; i<raw.length; i++) {
|
||||
int unsignedValue = raw[i];
|
||||
if (unsignedValue < 0) {
|
||||
unsignedValue += 256;
|
||||
}
|
||||
x = (x << 8) + unsignedValue;
|
||||
availableBits += 8;
|
||||
if (availableBits >= 12) {
|
||||
// take top six
|
||||
int highcode = (x >> (availableBits - 6)) & 0x3F;
|
||||
int highIndex = codeIndex((byte) (highcode));
|
||||
// take bottom six
|
||||
int lowcode = (x >> (availableBits - 12)) & 0x3F;
|
||||
int lowIndex = codeIndex((byte) (lowcode));
|
||||
// special case at end of transmission on uneven boundaries:
|
||||
if ((highIndex >= 0) && (lowIndex >= 0)) {
|
||||
byte decoded = (byte) ((highIndex << 4) + lowIndex);
|
||||
rval = ByteUtil.concat(rval, decoded);
|
||||
/*
|
||||
Log.d(TAG,String.format("i=%d,x=0x%08X,0x%02X->0x%02X, 0x%02X->0x%02X, result: 0x%02X, %d bits remaining, errors %d, bytes remaining: %s",
|
||||
i,x,highcode,highIndex, lowcode, lowIndex,decoded,availableBits,codingErrors,ByteUtil.shortHexString(ByteUtil.substring(raw,i+1,raw.length-i-1))));
|
||||
*/
|
||||
} else {
|
||||
//Log.d(TAG,String.format("i=%d,x=%08X, coding error: highcode=0x%02X, lowcode=0x%02X, %d bits remaining",i,x,highcode,lowcode,availableBits));
|
||||
codingErrors++;
|
||||
}
|
||||
|
||||
availableBits -= 12;
|
||||
x = x & (0x0000ffff >> (16 - availableBits));
|
||||
} else {
|
||||
//Log.d(TAG,String.format("i=%d, skip: x=0x%08X, available bits %d",i,x,availableBits));
|
||||
}
|
||||
}
|
||||
if (availableBits !=0) {
|
||||
if ((availableBits == 4) && (x == 0x05)) {
|
||||
// normal end
|
||||
} else {
|
||||
Log.e(TAG, "decode4b6b: failed clean decode -- extra bits available (not marker)(" + availableBits + ")");
|
||||
codingErrors++;
|
||||
}
|
||||
} else {
|
||||
// also normal end.
|
||||
}
|
||||
if (codingErrors>0) {
|
||||
Log.e(TAG, "decode4b6b: "+codingErrors+" coding errors encountered.");
|
||||
throw new NumberFormatException();
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
public static String toHexString(byte[] array) {
|
||||
return toHexString(array, 0, array.length);
|
||||
}
|
||||
|
||||
private final static char[] HEX_DIGITS = {
|
||||
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F'
|
||||
};
|
||||
|
||||
public static String toHexString(byte[] array, int offset, int length) {
|
||||
char[] buf = new char[length * 2];
|
||||
|
||||
int bufIndex = 0;
|
||||
for (int i = offset; i < offset + length; i++) {
|
||||
byte b = array[i];
|
||||
buf[bufIndex++] = HEX_DIGITS[(b >>> 4) & 0x0F];
|
||||
buf[bufIndex++] = HEX_DIGITS[b & 0x0F];
|
||||
}
|
||||
|
||||
return new String(buf);
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,24 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.CRC;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/22/16.
|
||||
*/
|
||||
|
||||
public class RadioPacket {
|
||||
protected byte[] pkt;
|
||||
public RadioPacket(byte[] pkt) {
|
||||
this.pkt = pkt;
|
||||
}
|
||||
public byte[] getRaw() {
|
||||
return pkt;
|
||||
}
|
||||
public byte[] getEncoded() {
|
||||
byte[] withCRC = ByteUtil.concat(pkt, CRC.crc8(pkt));
|
||||
byte[] encoded = RFTools.encode4b6b(withCRC);
|
||||
byte[] withNullTerm = ByteUtil.concat(encoded,(byte)0);
|
||||
return withNullTerm;
|
||||
}
|
||||
}
|
|
@ -1,67 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.CRC;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/30/16.
|
||||
*/
|
||||
public class RadioResponse {
|
||||
private static final String TAG = "RadioResponse";
|
||||
public boolean decodedOK = false;
|
||||
public int rssi;
|
||||
public int responseNumber;
|
||||
public byte[] decodedPayload = new byte[0];
|
||||
public byte receivedCRC;
|
||||
|
||||
public RadioResponse() {
|
||||
}
|
||||
|
||||
public RadioResponse(byte[] rxData) {
|
||||
init(rxData);
|
||||
}
|
||||
|
||||
public boolean isValid() {
|
||||
if (!decodedOK) {
|
||||
return false;
|
||||
}
|
||||
if (decodedPayload != null) {
|
||||
if (receivedCRC == CRC.crc8(decodedPayload)) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public void init(byte[] rxData) {
|
||||
if (rxData == null) {
|
||||
return;
|
||||
}
|
||||
if (rxData.length < 3) {
|
||||
// This does not look like something valid heard from a RileyLink device
|
||||
return;
|
||||
}
|
||||
rssi = rxData[0];
|
||||
responseNumber = rxData[1];
|
||||
byte[] encodedPayload = ByteUtil.substring(rxData, 2, rxData.length - 2);
|
||||
try {
|
||||
byte[] decodeThis = RFTools.decode4b6b(encodedPayload);
|
||||
decodedOK = true;
|
||||
decodedPayload = ByteUtil.substring(decodeThis, 0, decodeThis.length - 1);
|
||||
byte calculatedCRC = CRC.crc8(decodedPayload);
|
||||
receivedCRC = decodeThis[decodeThis.length - 1];
|
||||
if (receivedCRC != calculatedCRC) {
|
||||
Log.e(TAG, String.format("RadioResponse: CRC mismatch, calculated 0x%02x, received 0x%02x", calculatedCRC, receivedCRC));
|
||||
}
|
||||
} catch (NumberFormatException e) {
|
||||
decodedOK = false;
|
||||
Log.e(TAG, "Failed to decode radio data: " + ByteUtil.shortHexString(encodedPayload));
|
||||
}
|
||||
}
|
||||
|
||||
public byte[] getPayload() {
|
||||
return decodedPayload;
|
||||
}
|
||||
}
|
|
@ -1,428 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE;
|
||||
|
||||
import android.bluetooth.BluetoothAdapter;
|
||||
import android.bluetooth.BluetoothDevice;
|
||||
import android.bluetooth.BluetoothGatt;
|
||||
import android.bluetooth.BluetoothGattCallback;
|
||||
import android.bluetooth.BluetoothGattCharacteristic;
|
||||
import android.bluetooth.BluetoothGattDescriptor;
|
||||
import android.bluetooth.BluetoothGattService;
|
||||
import android.bluetooth.BluetoothManager;
|
||||
import android.bluetooth.BluetoothProfile;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.os.SystemClock;
|
||||
import android.support.v4.content.LocalBroadcastManager;
|
||||
import android.util.Log;
|
||||
import android.widget.Toast;
|
||||
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.BLECommOperation;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.BLECommOperationResult;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.CharacteristicReadOperation;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.CharacteristicWriteOperation;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.BLECommOperations.DescriptorWriteOperation;
|
||||
import com.gxwtech.roundtrip2.util.HexDump;
|
||||
import com.gxwtech.roundtrip2.util.ThreadUtil;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.UUID;
|
||||
import java.util.concurrent.Semaphore;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/26/16.
|
||||
*/
|
||||
public class RileyLinkBLE {
|
||||
private static final String TAG = "RileyLinkBLE";
|
||||
public boolean gattDebugEnabled = true;
|
||||
|
||||
private BluetoothAdapter bluetoothAdapter;
|
||||
private BluetoothGattCallback bluetoothGattCallback;
|
||||
|
||||
private final Context context;
|
||||
private BluetoothManager bluetoothManager;
|
||||
|
||||
private BluetoothDevice rileyLinkDevice;
|
||||
private BluetoothGatt bluetoothConnectionGatt = null;
|
||||
|
||||
private BLECommOperation mCurrentOperation;
|
||||
private Semaphore gattOperationSema = new Semaphore(1,true);
|
||||
|
||||
private Runnable radioResponseCountNotified;
|
||||
|
||||
private boolean mIsConnected = false;
|
||||
|
||||
public RileyLinkBLE(final Context context) {
|
||||
this.context = context;
|
||||
this.bluetoothManager = (BluetoothManager) context.getSystemService(Context.BLUETOOTH_SERVICE);
|
||||
this.bluetoothAdapter = bluetoothManager.getAdapter();
|
||||
|
||||
bluetoothGattCallback = new BluetoothGattCallback() {
|
||||
|
||||
@Override
|
||||
public void onCharacteristicChanged(final BluetoothGatt gatt, final BluetoothGattCharacteristic characteristic) {
|
||||
super.onCharacteristicChanged(gatt, characteristic);
|
||||
if (gattDebugEnabled) {
|
||||
Log.v(TAG, ThreadUtil.sig() + "onCharacteristicChanged " + GattAttributes.lookup(characteristic.getUuid()) + " " + HexDump.toHexString(characteristic.getValue()));
|
||||
if (characteristic.getUuid().equals(UUID.fromString(GattAttributes.CHARA_RADIO_RESPONSE_COUNT))) {
|
||||
Log.d(TAG, "Response Count is " + HexDump.toHexString(characteristic.getValue()));
|
||||
}
|
||||
}
|
||||
if (radioResponseCountNotified != null) {
|
||||
radioResponseCountNotified.run();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onCharacteristicRead(final BluetoothGatt gatt, final BluetoothGattCharacteristic characteristic, int status) {
|
||||
super.onCharacteristicRead(gatt, characteristic, status);
|
||||
|
||||
|
||||
final String statusMessage = getGattStatusMessage(status);
|
||||
if (gattDebugEnabled) {
|
||||
Log.v(TAG, ThreadUtil.sig() + "onCharacteristicRead (" + GattAttributes.lookup(characteristic.getUuid()) + ") "
|
||||
+ statusMessage + ":" + HexDump.toHexString(characteristic.getValue()));
|
||||
}
|
||||
mCurrentOperation.gattOperationCompletionCallback(characteristic.getUuid(),characteristic.getValue());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onCharacteristicWrite(final BluetoothGatt gatt, final BluetoothGattCharacteristic characteristic, int status) {
|
||||
super.onCharacteristicWrite(gatt, characteristic, status);
|
||||
|
||||
final String uuidString = GattAttributes.lookup(characteristic.getUuid());
|
||||
if (gattDebugEnabled) {
|
||||
Log.v(TAG, ThreadUtil.sig() + "onCharacteristicWrite " + getGattStatusMessage(status) + " " + uuidString + " " + HexDump.toHexString(characteristic.getValue()));
|
||||
}
|
||||
mCurrentOperation.gattOperationCompletionCallback(characteristic.getUuid(),characteristic.getValue());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onConnectionStateChange(final BluetoothGatt gatt, final int status, final int newState) {
|
||||
super.onConnectionStateChange(gatt, status, newState);
|
||||
|
||||
// https://github.com/NordicSemiconductor/puck-central-android/blob/master/PuckCentral/app/src/main/java/no/nordicsemi/puckcentral/bluetooth/gatt/GattManager.java#L117
|
||||
if (status == 133) {
|
||||
Log.e(TAG, "Got the status 133 bug, closing gatt");
|
||||
disconnect();
|
||||
return;
|
||||
}
|
||||
|
||||
if (gattDebugEnabled) {
|
||||
final String stateMessage;
|
||||
if (newState == BluetoothProfile.STATE_CONNECTED) {
|
||||
stateMessage = "CONNECTED";
|
||||
} else if (newState == BluetoothProfile.STATE_CONNECTING) {
|
||||
stateMessage = "CONNECTING";
|
||||
} else if (newState == BluetoothProfile.STATE_DISCONNECTED) {
|
||||
stateMessage = "DISCONNECTED";
|
||||
} else if (newState == BluetoothProfile.STATE_DISCONNECTING) {
|
||||
stateMessage = "DISCONNECTING";
|
||||
} else {
|
||||
stateMessage = "UNKNOWN (" + newState + ")";
|
||||
}
|
||||
|
||||
Log.w(TAG, "onConnectionStateChange " + getGattStatusMessage(status) + " " + stateMessage);
|
||||
}
|
||||
|
||||
if (status == BluetoothGatt.GATT_SUCCESS && newState == BluetoothProfile.STATE_CONNECTED) {
|
||||
LocalBroadcastManager.getInstance(context).sendBroadcast(new Intent(RT2Const.serviceLocal.bluetooth_connected));
|
||||
} else {
|
||||
LocalBroadcastManager.getInstance(context).sendBroadcast(new Intent(RT2Const.serviceLocal.bluetooth_disconnected));
|
||||
disconnect();
|
||||
Log.w(TAG, "Cannot establish Bluetooth connection.");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void onDescriptorWrite(BluetoothGatt gatt, BluetoothGattDescriptor descriptor, int status) {
|
||||
super.onDescriptorWrite(gatt, descriptor, status);
|
||||
if (gattDebugEnabled) {
|
||||
Log.w(TAG, "onDescriptorWrite "
|
||||
+ GattAttributes.lookup(descriptor.getUuid()) + " "
|
||||
+ getGattStatusMessage(status)
|
||||
+ " written: " + HexDump.toHexString(descriptor.getValue()));
|
||||
}
|
||||
mCurrentOperation.gattOperationCompletionCallback(descriptor.getUuid(),descriptor.getValue());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDescriptorRead(BluetoothGatt gatt, BluetoothGattDescriptor descriptor, int status) {
|
||||
super.onDescriptorRead(gatt, descriptor, status);
|
||||
mCurrentOperation.gattOperationCompletionCallback(descriptor.getUuid(),descriptor.getValue());
|
||||
if (gattDebugEnabled) {
|
||||
Log.w(TAG, "onDescriptorRead " + getGattStatusMessage(status) + " status " + descriptor);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onMtuChanged(BluetoothGatt gatt, int mtu, int status) {
|
||||
super.onMtuChanged(gatt, mtu, status);
|
||||
if (gattDebugEnabled) {
|
||||
Log.w(TAG, "onMtuChanged " + mtu + " status " + status);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onReadRemoteRssi(final BluetoothGatt gatt, int rssi, int status) {
|
||||
super.onReadRemoteRssi(gatt, rssi, status);
|
||||
if (gattDebugEnabled) {
|
||||
Log.w(TAG, "onReadRemoteRssi " + getGattStatusMessage(status) + ": " + rssi);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onReliableWriteCompleted(BluetoothGatt gatt, int status) {
|
||||
super.onReliableWriteCompleted(gatt, status);
|
||||
if (gattDebugEnabled) {
|
||||
Log.w(TAG, "onReliableWriteCompleted status " + status);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onServicesDiscovered(final BluetoothGatt gatt, int status) {
|
||||
super.onServicesDiscovered(gatt, status);
|
||||
|
||||
if (status == BluetoothGatt.GATT_SUCCESS) {
|
||||
final List<BluetoothGattService> services = gatt.getServices();
|
||||
// TODO: in here, we need to determine if this Bluetooth device is a RileyLink with appropriate software (subg_rfspy) and set mIsConnected if the GATT is ok.
|
||||
for (BluetoothGattService service : services) {
|
||||
final List<BluetoothGattCharacteristic> characteristics = service.getCharacteristics();
|
||||
|
||||
final UUID uuidService = service.getUuid();
|
||||
final String uuidServiceString = uuidService.toString();
|
||||
if (gattDebugEnabled) {
|
||||
String debugString = "Found service: " + GattAttributes.lookup(uuidServiceString, "Unknown device") + " (" + uuidServiceString + ")";
|
||||
|
||||
for (BluetoothGattCharacteristic character : characteristics) {
|
||||
final String uuidCharacteristicString = character.getUuid().toString();
|
||||
debugString += " - " + GattAttributes.lookup(uuidCharacteristicString);
|
||||
}
|
||||
Log.w(TAG, debugString);
|
||||
}
|
||||
}
|
||||
if (gattDebugEnabled) {
|
||||
Log.w(TAG, "onServicesDiscovered " + getGattStatusMessage(status));
|
||||
}
|
||||
mIsConnected = true;
|
||||
Intent intent = new Intent(RT2Const.serviceLocal.BLE_services_discovered);
|
||||
LocalBroadcastManager.getInstance(context).sendBroadcast(intent);
|
||||
} else {
|
||||
Log.e(TAG, "onServicesDiscovered " + getGattStatusMessage(status));
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
public void registerRadioResponseCountNotification(Runnable notifier) {
|
||||
radioResponseCountNotified = notifier;
|
||||
}
|
||||
|
||||
public boolean isConnected() {
|
||||
return mIsConnected;
|
||||
}
|
||||
|
||||
public void discoverServices() {
|
||||
if (bluetoothConnectionGatt.discoverServices()) {
|
||||
Log.w(TAG, "Starting to discover GATT Services.");
|
||||
|
||||
} else {
|
||||
Log.e(TAG, "Cannot discover GATT Services.");
|
||||
}
|
||||
}
|
||||
|
||||
public boolean enableNotifications() {
|
||||
BLECommOperationResult result = setNotification_blocking(
|
||||
UUID.fromString(GattAttributes.SERVICE_RADIO),
|
||||
UUID.fromString(GattAttributes.CHARA_RADIO_RESPONSE_COUNT));
|
||||
if (result.resultCode != BLECommOperationResult.RESULT_SUCCESS) {
|
||||
Log.e(TAG, "Error setting response count notification");
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public void findRileyLink(String RileyLinkAddress) {
|
||||
Log.d(TAG,"Rileylink address: " + RileyLinkAddress);
|
||||
// Must verify that this is a valid MAC, or crash.
|
||||
|
||||
rileyLinkDevice = bluetoothAdapter.getRemoteDevice(RileyLinkAddress);
|
||||
// if this succeeds, we get a connection state change callback?
|
||||
connectGatt();
|
||||
}
|
||||
|
||||
// This function must be run on UI thread.
|
||||
public void connectGatt() {
|
||||
bluetoothConnectionGatt = rileyLinkDevice.connectGatt(context, true, bluetoothGattCallback);
|
||||
if (bluetoothConnectionGatt == null) {
|
||||
Log.e(TAG,"Failed to connect to Bluetooth Low Energy device at "+bluetoothAdapter.getAddress());
|
||||
Toast.makeText(context, "No Rileylink at " + bluetoothAdapter.getAddress(), Toast.LENGTH_SHORT).show();
|
||||
} else {
|
||||
if (gattDebugEnabled) {
|
||||
Log.d(TAG, "Gatt Connected?");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void disconnect() {
|
||||
mIsConnected = false;
|
||||
Log.w(TAG, "Closing GATT connection");
|
||||
// Close old conenction
|
||||
if (bluetoothConnectionGatt != null) {
|
||||
// Not sure if to disconnect or to close first..
|
||||
bluetoothConnectionGatt.disconnect();
|
||||
bluetoothConnectionGatt.close();
|
||||
bluetoothConnectionGatt = null;
|
||||
}
|
||||
}
|
||||
|
||||
public BLECommOperationResult setNotification_blocking(UUID serviceUUID, UUID charaUUID) {
|
||||
BLECommOperationResult rval = new BLECommOperationResult();
|
||||
if (bluetoothConnectionGatt != null) {
|
||||
|
||||
try {
|
||||
gattOperationSema.acquire();
|
||||
SystemClock.sleep(1); // attempting to yield thread, to make sequence of events easier to follow
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG, "setNotification_blocking: interrupted waiting for gattOperationSema");
|
||||
return rval;
|
||||
}
|
||||
if (mCurrentOperation != null) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_BUSY;
|
||||
} else {
|
||||
if (bluetoothConnectionGatt.getService(serviceUUID) == null) {
|
||||
// Catch if the service is not supported by the BLE device
|
||||
rval.resultCode = BLECommOperationResult.RESULT_NONE;
|
||||
Log.e(TAG, "BT Device not supported");
|
||||
// TODO: 11/07/2016 UI update for user
|
||||
} else {
|
||||
BluetoothGattCharacteristic chara = bluetoothConnectionGatt.getService(serviceUUID).getCharacteristic(charaUUID);
|
||||
// Tell Android that we want the notifications
|
||||
bluetoothConnectionGatt.setCharacteristicNotification(chara, true);
|
||||
List<BluetoothGattDescriptor> list = chara.getDescriptors();
|
||||
if (gattDebugEnabled) {
|
||||
for (int i = 0; i < list.size(); i++) {
|
||||
Log.d(TAG, "Found descriptor: " + list.get(i).toString());
|
||||
}
|
||||
}
|
||||
BluetoothGattDescriptor descr = list.get(0);
|
||||
// Tell the remote device to send the notifications
|
||||
mCurrentOperation = new DescriptorWriteOperation(bluetoothConnectionGatt, descr, BluetoothGattDescriptor.ENABLE_NOTIFICATION_VALUE);
|
||||
mCurrentOperation.execute(this);
|
||||
if (mCurrentOperation.timedOut) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_TIMEOUT;
|
||||
} else if (mCurrentOperation.interrupted) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_INTERRUPTED;
|
||||
} else {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_SUCCESS;
|
||||
}
|
||||
}
|
||||
mCurrentOperation = null;
|
||||
gattOperationSema.release();
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG,"setNotification_blocking: not configured!");
|
||||
rval.resultCode = BLECommOperationResult.RESULT_NOT_CONFIGURED;
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
// call from main
|
||||
public BLECommOperationResult writeCharacteristic_blocking(UUID serviceUUID, UUID charaUUID, byte[] value) {
|
||||
BLECommOperationResult rval = new BLECommOperationResult();
|
||||
if (bluetoothConnectionGatt != null) {
|
||||
rval.value = value;
|
||||
try {
|
||||
gattOperationSema.acquire();
|
||||
SystemClock.sleep(1); // attempting to yield thread, to make sequence of events easier to follow
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG,"writeCharacteristic_blocking: interrupted waiting for gattOperationSema");
|
||||
return rval;
|
||||
}
|
||||
if (mCurrentOperation != null) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_BUSY;
|
||||
} else {
|
||||
if (bluetoothConnectionGatt.getService(serviceUUID) == null) {
|
||||
// Catch if the service is not supported by the BLE device
|
||||
// GGW: Tue Jul 12 01:14:01 UTC 2016: This can also happen if the
|
||||
// app that created the bluetoothConnectionGatt has been destroyed/created,
|
||||
// e.g. when the user switches from portrait to landscape.
|
||||
rval.resultCode = BLECommOperationResult.RESULT_NONE;
|
||||
Log.e(TAG, "BT Device not supported");
|
||||
// TODO: 11/07/2016 UI update for user
|
||||
} else {
|
||||
BluetoothGattCharacteristic chara = bluetoothConnectionGatt.getService(serviceUUID).getCharacteristic(charaUUID);
|
||||
mCurrentOperation = new CharacteristicWriteOperation(bluetoothConnectionGatt, chara, value);
|
||||
mCurrentOperation.execute(this);
|
||||
if (mCurrentOperation.timedOut) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_TIMEOUT;
|
||||
} else if (mCurrentOperation.interrupted) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_INTERRUPTED;
|
||||
} else {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_SUCCESS;
|
||||
}
|
||||
}
|
||||
mCurrentOperation = null;
|
||||
gattOperationSema.release();
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG,"writeCharacteristic_blocking: not configured!");
|
||||
rval.resultCode = BLECommOperationResult.RESULT_NOT_CONFIGURED;
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
public BLECommOperationResult readCharacteristic_blocking(UUID serviceUUID, UUID charaUUID) {
|
||||
BLECommOperationResult rval = new BLECommOperationResult();
|
||||
if (bluetoothConnectionGatt != null) {
|
||||
try {
|
||||
gattOperationSema.acquire();
|
||||
SystemClock.sleep(1); // attempting to yield thread, to make sequence of events easier to follow
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG, "readCharacteristic_blocking: Interrupted waiting for gattOperationSema");
|
||||
return rval;
|
||||
}
|
||||
if (mCurrentOperation != null) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_BUSY;
|
||||
} else {
|
||||
BluetoothGattCharacteristic chara = bluetoothConnectionGatt.getService(serviceUUID).getCharacteristic(charaUUID);
|
||||
mCurrentOperation = new CharacteristicReadOperation(bluetoothConnectionGatt, chara);
|
||||
mCurrentOperation.execute(this);
|
||||
if (mCurrentOperation.timedOut) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_TIMEOUT;
|
||||
} else if (mCurrentOperation.interrupted) {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_INTERRUPTED;
|
||||
} else {
|
||||
rval.resultCode = BLECommOperationResult.RESULT_SUCCESS;
|
||||
rval.value = mCurrentOperation.getValue();
|
||||
}
|
||||
}
|
||||
mCurrentOperation = null;
|
||||
gattOperationSema.release();
|
||||
} else {
|
||||
Log.e(TAG,"readCharacteristic_blocking: not configured!");
|
||||
rval.resultCode = BLECommOperationResult.RESULT_NOT_CONFIGURED;
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
private String getGattStatusMessage(final int status) {
|
||||
final String statusMessage;
|
||||
if (status == BluetoothGatt.GATT_SUCCESS) {
|
||||
statusMessage = "SUCCESS";
|
||||
} else if (status == BluetoothGatt.GATT_FAILURE) {
|
||||
statusMessage = "FAILED";
|
||||
} else if (status == BluetoothGatt.GATT_WRITE_NOT_PERMITTED) {
|
||||
statusMessage = "NOT PERMITTED";
|
||||
} else if (status == 133) {
|
||||
statusMessage = "Found the strange 133 bug";
|
||||
} else {
|
||||
statusMessage = "UNKNOWN (" + status + ")";
|
||||
}
|
||||
|
||||
return statusMessage;
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,638 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService;
|
||||
|
||||
import android.app.Service;
|
||||
import android.bluetooth.BluetoothAdapter;
|
||||
import android.bluetooth.BluetoothManager;
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.content.SharedPreferences;
|
||||
import android.content.res.Configuration;
|
||||
import android.os.Bundle;
|
||||
import android.os.IBinder;
|
||||
import android.os.Message;
|
||||
import android.os.Parcel;
|
||||
import android.os.PowerManager;
|
||||
import android.preference.PreferenceManager;
|
||||
import android.support.annotation.Nullable;
|
||||
import android.support.v4.content.LocalBroadcastManager;
|
||||
import android.util.Log;
|
||||
import android.widget.Toast;
|
||||
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLink.PumpManager;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RFSpy;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLinkBLE.RileyLinkBLE;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.DiscoverGattServicesTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.FetchPumpHistoryTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.InitializePumpManagerTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.ReadBolusWizardCarbProfileTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.ReadISFProfileTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.ReadPumpClockTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.RetrieveHistoryPageTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.ServiceTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.ServiceTaskExecutor;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.UpdatePumpStatusTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.Tasks.WakeAndTuneTask;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.ISFTable;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.Page;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.PumpHistoryManager;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.TimeFormat;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceNotification;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
|
||||
/**
|
||||
* RoundtripService is intended to stay running when the gui-app is closed.
|
||||
*/
|
||||
public class RoundtripService extends Service {
|
||||
private static final String TAG="RoundtripService";
|
||||
private static RoundtripService instance;
|
||||
private static final String WAKELOCKNAME = "com.gxwtech.roundtrip2.RoundtripServiceWakeLock";
|
||||
private static volatile PowerManager.WakeLock lockStatic = null;
|
||||
|
||||
private boolean needBluetoothPermission = true;
|
||||
|
||||
private BroadcastReceiver mBroadcastReceiver;
|
||||
private Context mContext;
|
||||
private RoundtripServiceIPCConnection serviceConnection;
|
||||
private BluetoothManager mBluetoothManager;
|
||||
private BluetoothAdapter mBluetoothAdapter;
|
||||
|
||||
|
||||
// saved settings
|
||||
|
||||
public SharedPreferences sharedPref;
|
||||
private String pumpIDString;
|
||||
private byte[] pumpIDBytes;
|
||||
private String mRileylinkAddress;
|
||||
|
||||
// cache of most recently received set of pump history pages. Probably shouldn't be here.
|
||||
ArrayList<Page> mHistoryPages;
|
||||
PumpHistoryManager pumpHistoryManager;
|
||||
|
||||
|
||||
// Our hardware/software connection
|
||||
public RileyLinkBLE rileyLinkBLE; // android-bluetooth management
|
||||
private RFSpy rfspy; // interface for 916MHz radio.
|
||||
public PumpManager pumpManager; // interface to Minimed
|
||||
private static ServiceTask currentTask = null;
|
||||
|
||||
public RoundtripService() {
|
||||
super();
|
||||
instance = this;
|
||||
Log.d(TAG, "RoundtripService newly constructed");
|
||||
}
|
||||
|
||||
public static RoundtripService getInstance() {
|
||||
return instance;
|
||||
}
|
||||
|
||||
public void onCreate() {
|
||||
super.onCreate();
|
||||
Log.d(TAG, "onCreate");
|
||||
mContext = getApplicationContext();
|
||||
serviceConnection = new RoundtripServiceIPCConnection(mContext);
|
||||
|
||||
//sharedPref = mContext.getSharedPreferences(RT2Const.serviceLocal.sharedPreferencesKey, Context.MODE_PRIVATE);
|
||||
sharedPref = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
|
||||
|
||||
// get most recently used pumpID
|
||||
pumpIDString = sharedPref.getString(RT2Const.serviceLocal.pumpIDKey,"000000");
|
||||
pumpIDBytes = ByteUtil.fromHexString(pumpIDString);
|
||||
if (pumpIDBytes == null) {
|
||||
Log.e(TAG,"Invalid pump ID? " + ByteUtil.shortHexString(pumpIDBytes));
|
||||
pumpIDBytes = new byte[] {0,0,0};
|
||||
pumpIDString = "000000";
|
||||
}
|
||||
if (pumpIDBytes.length != 3) {
|
||||
Log.e(TAG,"Invalid pump ID? " + ByteUtil.shortHexString(pumpIDBytes));
|
||||
pumpIDBytes = new byte[] {0,0,0};
|
||||
pumpIDString = "000000";
|
||||
}
|
||||
if (pumpIDString.equals("000000")) {
|
||||
Log.e(TAG,"Using pump ID "+pumpIDString);
|
||||
} else {
|
||||
Log.i(TAG,"Using pump ID "+pumpIDString);
|
||||
}
|
||||
|
||||
// get most recently used RileyLink address
|
||||
mRileylinkAddress = sharedPref.getString(RT2Const.serviceLocal.rileylinkAddressKey,"");
|
||||
|
||||
pumpHistoryManager = new PumpHistoryManager(getApplicationContext());
|
||||
rileyLinkBLE = new RileyLinkBLE(this);
|
||||
rfspy = new RFSpy(mContext,rileyLinkBLE);
|
||||
rfspy.startReader();
|
||||
pumpManager = new PumpManager(mContext,rfspy,pumpIDBytes);
|
||||
|
||||
mBroadcastReceiver = new BroadcastReceiver() {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
/* here we can listen for local broadcasts, then send ourselves
|
||||
* a specific intent to deal with them, if we wish
|
||||
*/
|
||||
if (intent == null) {
|
||||
Log.e(TAG,"onReceive: received null intent");
|
||||
} else {
|
||||
String action = intent.getAction();
|
||||
if (action == null) {
|
||||
Log.e(TAG,"onReceive: null action");
|
||||
} else {
|
||||
if (action.equals(RT2Const.serviceLocal.bluetooth_connected)) {
|
||||
Log.w(TAG,"serviceLocal.bluetooth_connected");
|
||||
serviceConnection.sendNotification(new ServiceNotification(RT2Const.IPC.MSG_note_FindingRileyLink),null);
|
||||
ServiceTaskExecutor.startTask(new DiscoverGattServicesTask());
|
||||
// If this is successful,
|
||||
// We will get a broadcast of RT2Const.serviceLocal.BLE_services_discovered
|
||||
} else if (action.equals(RT2Const.serviceLocal.BLE_services_discovered)) {
|
||||
Log.w(TAG,"serviceLocal.BLE_services_discovered");
|
||||
serviceConnection.sendNotification(new ServiceNotification(RT2Const.IPC.MSG_note_WakingPump),null);
|
||||
rileyLinkBLE.enableNotifications();
|
||||
rfspy.startReader(); // call startReader from outside?
|
||||
ServiceTask task = new InitializePumpManagerTask();
|
||||
ServiceTaskExecutor.startTask(task);
|
||||
Log.i(TAG, "Announcing RileyLink open For business");
|
||||
} else if (action.equals(RT2Const.serviceLocal.ipcBound)) {
|
||||
// If we still need permission for bluetooth, ask now.
|
||||
if (needBluetoothPermission) {
|
||||
sendBLERequestForAccess();
|
||||
}
|
||||
|
||||
} else if (RT2Const.IPC.MSG_BLE_accessGranted.equals(action)) {
|
||||
//initializeLeAdapter();
|
||||
//BluetoothInit();
|
||||
} else if (RT2Const.IPC.MSG_BLE_accessDenied.equals(action)) {
|
||||
stopSelf(); // This will stop the service.
|
||||
} else if (action.equals(RT2Const.IPC.MSG_PUMP_tunePump)) {
|
||||
doTunePump();
|
||||
} else if (action.equals(RT2Const.IPC.MSG_PUMP_quickTune)) {
|
||||
doTunePump();
|
||||
} else if (action.equals(RT2Const.IPC.MSG_PUMP_fetchHistory)) {
|
||||
mHistoryPages = pumpManager.getAllHistoryPages();
|
||||
final boolean savePages = true;
|
||||
if (savePages) {
|
||||
for (int i = 0; i < mHistoryPages.size(); i++) {
|
||||
String filename = "PumpHistoryPage-" + i;
|
||||
Log.w(TAG, "Saving history page to file " + filename);
|
||||
FileOutputStream outputStream;
|
||||
try {
|
||||
outputStream = openFileOutput(filename, 0);
|
||||
byte[] rawData= mHistoryPages.get(i).getRawData();
|
||||
if (rawData != null) {
|
||||
outputStream.write(rawData);
|
||||
}
|
||||
outputStream.close();
|
||||
} catch (FileNotFoundException fnf) {
|
||||
fnf.printStackTrace();
|
||||
} catch (IOException ioe) {
|
||||
ioe.printStackTrace();
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
Message msg = Message.obtain(null, RT2Const.IPC.MSG_IPC, 0, 0);
|
||||
// Create a bundle with the data
|
||||
Bundle bundle = new Bundle();
|
||||
bundle.putString(RT2Const.IPC.messageKey, RT2Const.IPC.MSG_PUMP_history);
|
||||
ArrayList<Bundle> packedPages = new ArrayList<>();
|
||||
for (Page page : mHistoryPages) {
|
||||
packedPages.add(page.pack());
|
||||
}
|
||||
bundle.putParcelableArrayList(RT2Const.IPC.MSG_PUMP_history_key, packedPages);
|
||||
|
||||
// save it to SQL.
|
||||
pumpHistoryManager.clearDatabase();
|
||||
pumpHistoryManager.initFromPages(bundle);
|
||||
// write html page to documents folder
|
||||
pumpHistoryManager.writeHtmlPage();
|
||||
|
||||
// Set payload
|
||||
msg.setData(bundle);
|
||||
serviceConnection.sendMessage(msg,null/*broadcast*/);
|
||||
Log.d(TAG, "sendMessage: sent Full history report");
|
||||
} else if (RT2Const.IPC.MSG_PUMP_fetchSavedHistory.equals(action)) {
|
||||
Log.i(TAG,"Fetching saved history");
|
||||
FileInputStream inputStream;
|
||||
ArrayList<Page> storedHistoryPages = new ArrayList<>();
|
||||
for (int i = 0; i < 16; i++) {
|
||||
|
||||
String filename = "PumpHistoryPage-" + i;
|
||||
try {
|
||||
inputStream = openFileInput(filename);
|
||||
byte[] buffer = new byte[1024];
|
||||
int numRead = inputStream.read(buffer, 0, 1024);
|
||||
if (numRead == 1024) {
|
||||
Page p = new Page();
|
||||
//p.parseFrom(buffer, PumpModel.MM522);
|
||||
p.parseFrom(buffer, PumpModel.MM522);
|
||||
storedHistoryPages.add(p);
|
||||
} else {
|
||||
Log.e(TAG, filename + " error: short file");
|
||||
}
|
||||
} catch (FileNotFoundException fnf) {
|
||||
Log.e(TAG, "Failed to open " + filename + " for reading.");
|
||||
} catch (IOException e) {
|
||||
Log.e(TAG, "Failed to read " + filename);
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
mHistoryPages = storedHistoryPages;
|
||||
if (storedHistoryPages.isEmpty()) {
|
||||
Log.e(TAG, "No stored history pages loaded");
|
||||
} else {
|
||||
Message msg = Message.obtain(null, RT2Const.IPC.MSG_IPC, 0, 0);
|
||||
// Create a bundle with the data
|
||||
Bundle bundle = new Bundle();
|
||||
bundle.putString(RT2Const.IPC.messageKey, RT2Const.IPC.MSG_PUMP_history);
|
||||
ArrayList<Bundle> packedPages = new ArrayList<>();
|
||||
for (Page page : mHistoryPages) {
|
||||
packedPages.add(page.pack());
|
||||
}
|
||||
bundle.putParcelableArrayList(RT2Const.IPC.MSG_PUMP_history_key, packedPages);
|
||||
|
||||
// save it to SQL.
|
||||
pumpHistoryManager.clearDatabase();
|
||||
pumpHistoryManager.initFromPages(bundle);
|
||||
// write html page to documents folder
|
||||
pumpHistoryManager.writeHtmlPage();
|
||||
|
||||
// Set payload
|
||||
msg.setData(bundle);
|
||||
serviceConnection.sendMessage(msg,null/*broadcast*/);
|
||||
|
||||
}
|
||||
} else if (RT2Const.IPC.MSG_ServiceCommand.equals(action)) {
|
||||
Bundle bundle = intent.getBundleExtra(RT2Const.IPC.bundleKey);
|
||||
|
||||
handleIncomingServiceTransport(new ServiceTransport(bundle));
|
||||
} else if (RT2Const.serviceLocal.INTENT_sessionCompleted.equals(action)) {
|
||||
Bundle bundle = intent.getBundleExtra(RT2Const.IPC.bundleKey);
|
||||
if (bundle != null) {
|
||||
ServiceTransport transport = new ServiceTransport(bundle);
|
||||
serviceConnection.sendTransport(transport, transport.getSenderHashcode());
|
||||
} else {
|
||||
Log.e(TAG,"sessionCompleted: no bundle!");
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG, "Unhandled broadcast: action=" + action);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
IntentFilter intentFilter = new IntentFilter();
|
||||
intentFilter.addAction(RT2Const.serviceLocal.bluetooth_connected);
|
||||
intentFilter.addAction(RT2Const.serviceLocal.bluetooth_disconnected);
|
||||
intentFilter.addAction(RT2Const.serviceLocal.BLE_services_discovered);
|
||||
intentFilter.addAction(RT2Const.serviceLocal.ipcBound);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_BLE_accessGranted);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_BLE_accessDenied);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_BLE_useThisDevice);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_PUMP_tunePump);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_PUMP_fetchHistory);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_PUMP_useThisAddress);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_PUMP_fetchSavedHistory);
|
||||
intentFilter.addAction(RT2Const.IPC.MSG_ServiceCommand);
|
||||
intentFilter.addAction(RT2Const.serviceLocal.INTENT_sessionCompleted);
|
||||
|
||||
LocalBroadcastManager.getInstance(mContext).registerReceiver(mBroadcastReceiver, intentFilter);
|
||||
|
||||
Log.d(TAG, "onCreate(): It's ALIVE!");
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onUnbind(Intent intent) {
|
||||
Log.w(TAG,"onUnbind");
|
||||
return super.onUnbind(intent);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onRebind(Intent intent) {
|
||||
Log.w(TAG,"onRebind");
|
||||
super.onRebind(intent);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onConfigurationChanged(Configuration newConfig) {
|
||||
Log.w(TAG,"onConfigurationChanged");
|
||||
super.onConfigurationChanged(newConfig);
|
||||
}
|
||||
|
||||
@Nullable
|
||||
@Override
|
||||
public IBinder onBind(Intent intent) {
|
||||
return serviceConnection.doOnBind(intent);
|
||||
}
|
||||
|
||||
// Here is where the wake-lock begins:
|
||||
// We've received a service startCommand, we grab the lock.
|
||||
@Override
|
||||
public int onStartCommand(Intent intent, int flags, int startId) {
|
||||
Log.d(TAG, "onStartCommand");
|
||||
if (intent != null) {
|
||||
PowerManager.WakeLock lock = getLock(this.getApplicationContext());
|
||||
|
||||
if (!lock.isHeld() || (flags & START_FLAG_REDELIVERY) != 0) {
|
||||
lock.acquire();
|
||||
}
|
||||
|
||||
// This will end up running onHandleIntent
|
||||
super.onStartCommand(intent, flags, startId);
|
||||
} else {
|
||||
Log.e(TAG, "Received null intent?");
|
||||
}
|
||||
BluetoothInit(); // this kicks off our process of device discovery.
|
||||
return (START_REDELIVER_INTENT | START_STICKY);
|
||||
}
|
||||
|
||||
synchronized private static PowerManager.WakeLock getLock(Context context) {
|
||||
if (lockStatic == null) {
|
||||
PowerManager mgr =
|
||||
(PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
|
||||
lockStatic = mgr.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, WAKELOCKNAME);
|
||||
lockStatic.setReferenceCounted(true);
|
||||
}
|
||||
|
||||
return lockStatic;
|
||||
}
|
||||
|
||||
// FIXME: This needs to be run in a session so that is interruptable, has a separate thread, etc.
|
||||
private void doTunePump() {
|
||||
double lastGoodFrequency = sharedPref.getFloat(RT2Const.serviceLocal.prefsLastGoodPumpFrequency,(float)0.0);
|
||||
double newFrequency;
|
||||
if (lastGoodFrequency != 0.0) {
|
||||
Log.i(TAG,String.format("Checking for pump near last saved frequency of %.2fMHz",lastGoodFrequency));
|
||||
// we have an old frequency, so let's start there.
|
||||
newFrequency = pumpManager.quickTuneForPump(lastGoodFrequency);
|
||||
if (newFrequency == 0.0) {
|
||||
// quick scan failed to find pump. Try full scan
|
||||
Log.w(TAG,String.format("Failed to find pump near last saved frequency, doing full scan"));
|
||||
newFrequency = pumpManager.tuneForPump();
|
||||
}
|
||||
} else {
|
||||
Log.w(TAG,"No saved frequency for pump, doing full scan.");
|
||||
// we don't have a saved frequency, so do the full scan.
|
||||
newFrequency = pumpManager.tuneForPump();
|
||||
|
||||
}
|
||||
if ((newFrequency!=0.0) && (newFrequency != lastGoodFrequency)) {
|
||||
Log.i(TAG,String.format("Saving new pump frequency of %.2fMHz",newFrequency));
|
||||
SharedPreferences.Editor ed = sharedPref.edit();
|
||||
ed.putFloat(RT2Const.serviceLocal.prefsLastGoodPumpFrequency, (float)newFrequency);
|
||||
ed.apply();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDestroy() {
|
||||
super.onDestroy();
|
||||
Log.e(TAG, "I die! I die!");
|
||||
}
|
||||
|
||||
/* private functions */
|
||||
|
||||
void BluetoothInit() {
|
||||
if (mBluetoothManager == null) {
|
||||
mBluetoothManager = (BluetoothManager) getSystemService(Context.BLUETOOTH_SERVICE);
|
||||
if (mBluetoothManager == null) {
|
||||
Log.e(TAG, "Unable to initialize BluetoothManager.");
|
||||
}
|
||||
}
|
||||
// Ensures Bluetooth is available on the device and it is enabled. If not,
|
||||
// displays a dialog requesting user permission to enable Bluetooth.
|
||||
if ((mBluetoothAdapter==null) || (!mBluetoothAdapter.isEnabled())) {
|
||||
sendBLERequestForAccess();
|
||||
} else {
|
||||
needBluetoothPermission = false;
|
||||
initializeLeAdapter();
|
||||
}
|
||||
}
|
||||
|
||||
public boolean initializeLeAdapter() {
|
||||
Log.d(TAG,"initializeLeAdapter: attempting to get an adapter");
|
||||
mBluetoothAdapter = mBluetoothManager.getAdapter();
|
||||
if (mBluetoothAdapter == null) {
|
||||
Log.e(TAG, "Unable to obtain a BluetoothAdapter.");
|
||||
return false;
|
||||
} else if (!mBluetoothAdapter.isEnabled()) {
|
||||
// NOTE: This does not work!
|
||||
Log.e(TAG, "Bluetooth is not enabled.");
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
private void setPumpIDString(String idString) {
|
||||
if (idString.length() != 6) {
|
||||
Log.e(TAG,"setPumpIDString: invalid pump id string: " + idString);
|
||||
}
|
||||
pumpIDString = idString;
|
||||
pumpIDBytes = ByteUtil.fromHexString(pumpIDString);
|
||||
SharedPreferences prefs = mContext.getSharedPreferences(RT2Const.serviceLocal.sharedPreferencesKey, Context.MODE_PRIVATE);
|
||||
SharedPreferences.Editor editor = prefs.edit();
|
||||
editor.putString(RT2Const.serviceLocal.pumpIDKey,pumpIDString);
|
||||
editor.apply();
|
||||
Log.i(TAG,"setPumpIDString: saved pumpID "+pumpIDString);
|
||||
}
|
||||
|
||||
private void sendBLERequestForAccess() {
|
||||
//serviceConnection.sendMessage(RT2Const.IPC.MSG_BLE_requestAccess);
|
||||
}
|
||||
|
||||
private void reportPumpFound() {
|
||||
//serviceConnection.sendMessage(RT2Const.IPC.MSG_PUMP_pumpFound);
|
||||
}
|
||||
|
||||
public void setCurrentTask(ServiceTask task) {
|
||||
if (currentTask == null) {
|
||||
currentTask = task;
|
||||
} else {
|
||||
Log.e(TAG,"setCurrentTask: Cannot replace current task");
|
||||
}
|
||||
}
|
||||
|
||||
public void finishCurrentTask(ServiceTask task) {
|
||||
if (task != currentTask) {
|
||||
Log.e(TAG,"finishCurrentTask: task does not match");
|
||||
}
|
||||
// hack to force deep copy of transport contents
|
||||
ServiceTransport transport = task.getServiceTransport().clone();
|
||||
|
||||
if (transport.hasServiceResult()) {
|
||||
sendServiceTransportResponse(transport,transport.getServiceResult());
|
||||
}
|
||||
currentTask = null;
|
||||
}
|
||||
|
||||
private void handleIncomingServiceTransport(ServiceTransport serviceTransport) {
|
||||
if (serviceTransport.getServiceCommand().isPumpCommand()) {
|
||||
switch (serviceTransport.getOriginalCommandName()) {
|
||||
case "ReadPumpClock":
|
||||
ServiceTaskExecutor.startTask(new ReadPumpClockTask(serviceTransport));
|
||||
break;
|
||||
case "FetchPumpHistory":
|
||||
ServiceTaskExecutor.startTask(new FetchPumpHistoryTask(serviceTransport));
|
||||
break;
|
||||
case "RetrieveHistoryPage":
|
||||
ServiceTask task = new RetrieveHistoryPageTask(serviceTransport);
|
||||
ServiceTaskExecutor.startTask(task);
|
||||
break;
|
||||
case "ReadISFProfile":
|
||||
ServiceTaskExecutor.startTask(new ReadISFProfileTask(serviceTransport));
|
||||
/*
|
||||
ISFTable table = pumpManager.getPumpISFProfile();
|
||||
ServiceResult result = new ServiceResult();
|
||||
if (table.isValid()) {
|
||||
// convert from ISFTable to ISFProfile
|
||||
Bundle map = result.getMap();
|
||||
map.putIntArray("times", table.getTimes());
|
||||
map.putFloatArray("rates", table.getRates());
|
||||
map.putString("ValidDate", TimeFormat.standardFormatter().print(table.getValidDate()));
|
||||
result.setMap(map);
|
||||
result.setResultOK();
|
||||
}
|
||||
sendServiceTransportResponse(serviceTransport,result);
|
||||
*/
|
||||
break;
|
||||
case "ReadBolusWizardCarbProfile":
|
||||
ServiceTaskExecutor.startTask(new ReadBolusWizardCarbProfileTask());
|
||||
break;
|
||||
case "UpdatePumpStatus":
|
||||
ServiceTaskExecutor.startTask(new UpdatePumpStatusTask());
|
||||
break;
|
||||
case "WakeAndTune":
|
||||
ServiceTaskExecutor.startTask(new WakeAndTuneTask());
|
||||
default:
|
||||
Log.e(TAG,"Failed to handle pump command: " + serviceTransport.getOriginalCommandName());
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
switch (serviceTransport.getOriginalCommandName()) {
|
||||
case "SetPumpID":
|
||||
// This one is a command to RoundtripService, not to the PumpManager
|
||||
String pumpID = serviceTransport.getServiceCommand().getMap().getString("pumpID", "");
|
||||
ServiceResult result = new ServiceResult();
|
||||
if ((pumpID != null) && (pumpID.length() == 6)) {
|
||||
setPumpIDString(pumpID);
|
||||
result.setResultOK();
|
||||
} else {
|
||||
Log.e(TAG, "handleIncomingServiceTransport: SetPumpID bundle missing 'pumpID' value");
|
||||
result.setResultError(-1, "Invalid parameter (missing pumpID)");
|
||||
}
|
||||
sendServiceTransportResponse(serviceTransport, result);
|
||||
break;
|
||||
case "UseThisRileylink":
|
||||
// If we are not connected, connect using the given address.
|
||||
// If we are connected and the addresses differ, disconnect, connect to new.
|
||||
// If we are connected and the addresses are the same, ignore.
|
||||
String deviceAddress = serviceTransport.getServiceCommand().getMap().getString("rlAddress", "");
|
||||
if ("".equals(deviceAddress)) {
|
||||
Log.e(TAG, "handleIPCMessage: null RL address passed");
|
||||
} else {
|
||||
reconfigureRileylink(deviceAddress);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
Log.e(TAG, "handleIncomingServiceTransport: Failed to handle service command '" + serviceTransport.getOriginalCommandName() + "'");
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// returns true if our Rileylink configuration changed
|
||||
public boolean reconfigureRileylink(String deviceAddress) {
|
||||
if (rileyLinkBLE.isConnected()) {
|
||||
if (deviceAddress.equals(mRileylinkAddress)) {
|
||||
Log.i(TAG, "No change to RL address. Not reconnecting.");
|
||||
return false;
|
||||
} else {
|
||||
Log.w(TAG, "Disconnecting from old RL (" + mRileylinkAddress + "), reconnecting to new: " + deviceAddress);
|
||||
rileyLinkBLE.disconnect();
|
||||
// prolly need to shut down listening thread too?
|
||||
SharedPreferences.Editor ed = sharedPref.edit();
|
||||
ed.putString("rlAddress", deviceAddress);
|
||||
ed.apply();
|
||||
mRileylinkAddress = deviceAddress;
|
||||
rileyLinkBLE.findRileyLink(mRileylinkAddress);
|
||||
return true;
|
||||
}
|
||||
} else {
|
||||
Toast.makeText(mContext, "Using RL " + deviceAddress, Toast.LENGTH_SHORT).show();
|
||||
Log.d(TAG, "handleIPCMessage: Using RL " + deviceAddress);
|
||||
if (mBluetoothAdapter == null) {
|
||||
mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
|
||||
}
|
||||
if (mBluetoothAdapter != null) {
|
||||
if (mBluetoothAdapter.isEnabled()) {
|
||||
// FIXME: this may be a long running function:
|
||||
rileyLinkBLE.findRileyLink(deviceAddress);
|
||||
// If successful, we will get a broadcast from RileyLinkBLE: RT2Const.serviceLocal.bluetooth_connected
|
||||
return true;
|
||||
} else {
|
||||
Log.e(TAG, "Bluetooth is not enabled.");
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG, "Failed to get adapter");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void announceProgress(int progressPercent) {
|
||||
if (currentTask != null) {
|
||||
ServiceNotification note = new ServiceNotification(RT2Const.IPC.MSG_note_TaskProgress);
|
||||
note.getMap().putInt("progress",progressPercent);
|
||||
note.getMap().putString("task",currentTask.getServiceTransport().getOriginalCommandName());
|
||||
Integer senderHashcode = currentTask.getServiceTransport().getSenderHashcode();
|
||||
serviceConnection.sendNotification(note, senderHashcode);
|
||||
} else {
|
||||
Log.e(TAG,"announceProgress: No current task");
|
||||
}
|
||||
}
|
||||
|
||||
public void sendServiceTransportResponse(ServiceTransport transport, ServiceResult serviceResult) {
|
||||
// get the key (hashcode) of the client who requested this
|
||||
Integer clientHashcode = transport.getSenderHashcode();
|
||||
// make a new bundle to send as the message data
|
||||
transport.setServiceResult(serviceResult);
|
||||
transport.setTransportType(RT2Const.IPC.MSG_ServiceResult);
|
||||
serviceConnection.sendTransport(transport,clientHashcode);
|
||||
}
|
||||
|
||||
public boolean sendNotification(ServiceNotification notification, Integer clientHashcode) {
|
||||
return serviceConnection.sendNotification(notification, clientHashcode);
|
||||
}
|
||||
|
||||
public void saveHistoryPage(int pagenumber, Page page) {
|
||||
if ((page == null) || (page.getRawData() == null)) {
|
||||
return;
|
||||
}
|
||||
String filename = "history-" + pagenumber;
|
||||
FileOutputStream os;
|
||||
try {
|
||||
os = openFileOutput(filename, Context.MODE_PRIVATE);
|
||||
os.write(page.getRawData());
|
||||
os.close();
|
||||
} catch (FileNotFoundException e) {
|
||||
e.printStackTrace();
|
||||
} catch (IOException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,165 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.IBinder;
|
||||
import android.os.Message;
|
||||
import android.os.Messenger;
|
||||
import android.os.RemoteException;
|
||||
import android.support.v4.content.LocalBroadcastManager;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceNotification;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
import java.util.HashMap;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/11/16.
|
||||
*/
|
||||
public class RoundtripServiceIPCConnection {
|
||||
private static final String TAG = "RTServiceIPC";
|
||||
private Context context;
|
||||
//private ArrayList<Messenger> mClients = new ArrayList<>();
|
||||
private HashMap<Integer,Messenger> mClients = new HashMap<>();
|
||||
|
||||
private final Messenger mMessenger = new Messenger(new IncomingHandler());
|
||||
|
||||
public RoundtripServiceIPCConnection(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
class IncomingHandler extends Handler {
|
||||
@Override
|
||||
public void handleMessage(Message msg) {
|
||||
Log.d(TAG, "handleMessage: Received message " + msg);
|
||||
Bundle bundle = msg.getData();
|
||||
switch(msg.what) {
|
||||
// This just helps sub-divide the message processing
|
||||
case RT2Const.IPC.MSG_registerClient:
|
||||
// send a reply, to let them know we're listening.
|
||||
Message myReply = Message.obtain(null, RT2Const.IPC.MSG_clientRegistered,0,0);
|
||||
try {
|
||||
msg.replyTo.send(myReply);
|
||||
mClients.put(mClients.hashCode(),msg.replyTo);
|
||||
Log.v(TAG,"handleMessage: Registered client");
|
||||
} catch (RemoteException e) {
|
||||
// I guess they aren't registered after all...
|
||||
Log.e(TAG,"handleMessage: failed to send acknowledgement of registration");
|
||||
}
|
||||
|
||||
break;
|
||||
case RT2Const.IPC.MSG_unregisterClient:
|
||||
Log.v(TAG,"Unregistered client");
|
||||
mClients.remove(msg.replyTo.hashCode());
|
||||
break;
|
||||
case RT2Const.IPC.MSG_IPC:
|
||||
// As the current thread is likely a GUI thread from some app,
|
||||
// rebroadcast the message as a local item.
|
||||
// Convert from Message to Intent
|
||||
if (msg.replyTo != null) {
|
||||
try {
|
||||
ServiceTransport transport = new ServiceTransport(bundle);
|
||||
Log.d(TAG, "Service received IPC message" + transport.describeContentsShort());
|
||||
transport.setSenderHashcode(msg.replyTo.hashCode());
|
||||
Intent intent = new Intent(transport.getTransportType());
|
||||
intent.putExtra(RT2Const.IPC.bundleKey, transport.getMap());
|
||||
LocalBroadcastManager.getInstance(context).sendBroadcast(intent);
|
||||
} catch (IllegalArgumentException e) {
|
||||
// This can happen on screen tilts... what else is wrong?
|
||||
Log.e(TAG,"Malformed service bundle: " + bundle.toString());
|
||||
}
|
||||
}
|
||||
break;
|
||||
/*
|
||||
case RT2Const.MSG_ping:
|
||||
String hello = (String)bundle.get("key_hello");
|
||||
Toast.makeText(mContext,hello, Toast.LENGTH_SHORT).show();
|
||||
break;
|
||||
*/
|
||||
default:
|
||||
Log.e(TAG,"handleMessage: unknown 'what' in message: "+msg.what);
|
||||
super.handleMessage(msg);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public IBinder doOnBind(Intent intent) {
|
||||
Log.d(TAG, "onBind");
|
||||
LocalBroadcastManager.getInstance(context).sendBroadcast(new Intent(RT2Const.serviceLocal.ipcBound));
|
||||
return mMessenger.getBinder();
|
||||
}
|
||||
|
||||
public boolean sendNotification(ServiceNotification notification, Integer clientHashcode) {
|
||||
ServiceTransport transport = new ServiceTransport();
|
||||
transport.setServiceNotification(notification);
|
||||
transport.setTransportType(RT2Const.IPC.MSG_ServiceNotification);
|
||||
return sendTransport(transport,clientHashcode);
|
||||
}
|
||||
|
||||
// if clientHashcode is null, broadcast to all clients
|
||||
public boolean sendMessage(Message msg, Integer clientHashcode) {
|
||||
Messenger clientMessenger = null;
|
||||
|
||||
if (mClients.isEmpty()) {
|
||||
if (msg.what == RT2Const.IPC.MSG_IPC) {
|
||||
Log.e(TAG, "sendMessage: no clients, cannot send: " + msg.getData().getString(RT2Const.IPC.messageKey, "(unknown)"));
|
||||
} else {
|
||||
Log.e(TAG, "sendMessage: no clients, cannot send: what=" + msg.what);
|
||||
}
|
||||
} else {
|
||||
if (clientHashcode != null) {
|
||||
clientMessenger = mClients.get(clientHashcode);
|
||||
}
|
||||
try {
|
||||
if (clientMessenger != null) {
|
||||
// sending to just one client
|
||||
clientMessenger.send(msg);
|
||||
} else {
|
||||
// send to all clients
|
||||
for (Integer clientHash : mClients.keySet()) {
|
||||
Message m2 = Message.obtain(msg);
|
||||
mClients.get(clientHash).send(m2);
|
||||
}
|
||||
}
|
||||
} catch (RemoteException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
/*
|
||||
catch (IllegalStateException e) {
|
||||
// This happens every time we are in a bluetooth operation and the screen is turned.
|
||||
Log.e(TAG,"sendMessage: IllegalStateException");
|
||||
}
|
||||
*/
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/*
|
||||
public void sendMessage(String messageType) {
|
||||
Message msg = Message.obtain(null, RT2Const.IPC.MSG_IPC,0,0);
|
||||
// Create a bundle with the data
|
||||
Bundle bundle = new Bundle();
|
||||
bundle.putString(RT2Const.IPC.messageKey, messageType);
|
||||
|
||||
// Set payload
|
||||
msg.setData(bundle);
|
||||
sendMessage(msg, null);
|
||||
Log.d(TAG,"sendMessage: sent "+messageType);
|
||||
}
|
||||
*/
|
||||
|
||||
public boolean sendTransport(ServiceTransport transport, Integer clientHashcode) {
|
||||
Message msg = Message.obtain(null, RT2Const.IPC.MSG_IPC,0,0);
|
||||
// Set payload
|
||||
msg.setData(transport.getMap());
|
||||
Log.d(TAG,"Service sending message to " + String.valueOf(clientHashcode) + ": " + transport.describeContentsShort());
|
||||
if ((clientHashcode != null) && (clientHashcode == 0)) {
|
||||
return sendMessage(msg, null);
|
||||
}
|
||||
return sendMessage(msg, clientHashcode);
|
||||
}
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/9/16.
|
||||
*/
|
||||
public class DiscoverGattServicesTask extends ServiceTask {
|
||||
public DiscoverGattServicesTask() {}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
RoundtripService.getInstance().rileyLinkBLE.discoverServices();
|
||||
}
|
||||
}
|
|
@ -1,40 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.Page;
|
||||
import com.gxwtech.roundtrip2.ServiceData.FetchPumpHistoryResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.RetrieveHistoryPageResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
import java.net.MalformedURLException;
|
||||
import java.util.ArrayList;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/16/16.
|
||||
*/
|
||||
public class FetchPumpHistoryTask extends PumpTask {
|
||||
public FetchPumpHistoryTask() { }
|
||||
public FetchPumpHistoryTask(ServiceTransport transport) {
|
||||
super(transport);
|
||||
}
|
||||
private FetchPumpHistoryResult result = new FetchPumpHistoryResult();
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
ArrayList<Page> ra = new ArrayList<>();
|
||||
for (int i=0; i<16; i++) {
|
||||
Page page = RoundtripService.getInstance().pumpManager.getPumpHistoryPage(i);
|
||||
if (page != null) {
|
||||
ra.add(page);
|
||||
RoundtripService.getInstance().saveHistoryPage(i,page);
|
||||
}
|
||||
}
|
||||
|
||||
result.setMap(getServiceTransport().getServiceResult().getMap());
|
||||
result.setResultOK();
|
||||
result.setPageArray(ra);
|
||||
getServiceTransport().setServiceResult(result);
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,42 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.SharedPreferences;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLink.PumpManager;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceNotification;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/9/16.
|
||||
*
|
||||
* This class is intended to be run by the Service, for the Service.
|
||||
* Not intended for clients to run.
|
||||
*/
|
||||
public class InitializePumpManagerTask extends ServiceTask {
|
||||
private static final String TAG = "InitPumpManagerTask";
|
||||
public InitializePumpManagerTask() { super(); }
|
||||
public InitializePumpManagerTask(ServiceTransport transport) { super(transport); }
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
SharedPreferences sharedPref = RoundtripService.getInstance().getApplicationContext().getSharedPreferences(RT2Const.serviceLocal.sharedPreferencesKey, Context.MODE_PRIVATE);
|
||||
double lastGoodFrequency = sharedPref.getFloat(RT2Const.serviceLocal.prefsLastGoodPumpFrequency,(float)0.0);
|
||||
if (lastGoodFrequency != 0) {
|
||||
Log.i(TAG,String.format("Setting radio frequency to %.2fMHz",lastGoodFrequency));
|
||||
RoundtripService.getInstance().pumpManager.setRadioFrequencyForPump(lastGoodFrequency);
|
||||
}
|
||||
|
||||
PumpModel reportedPumpModel = RoundtripService.getInstance().pumpManager.getPumpModel();
|
||||
if (!reportedPumpModel.equals(PumpModel.UNSET)) {
|
||||
RoundtripService.getInstance().sendNotification(new ServiceNotification(RT2Const.IPC.MSG_PUMP_pumpFound),null);
|
||||
} else {
|
||||
RoundtripService.getInstance().sendNotification(new ServiceNotification(RT2Const.IPC.MSG_PUMP_pumpLost),null);
|
||||
}
|
||||
RoundtripService.getInstance().sendNotification(new ServiceNotification(RT2Const.IPC.MSG_note_Idle),null);
|
||||
}
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/10/16.
|
||||
*/
|
||||
public class PumpTask extends ServiceTask {
|
||||
public PumpTask() {super();}
|
||||
public PumpTask(ServiceTransport transport) {
|
||||
super(transport);
|
||||
}
|
||||
}
|
|
@ -1,24 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpMessage;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/10/16.
|
||||
*/
|
||||
public class ReadBolusWizardCarbProfileTask extends PumpTask {
|
||||
public ReadBolusWizardCarbProfileTask() { super(); }
|
||||
public ReadBolusWizardCarbProfileTask(ServiceTransport transport) {
|
||||
super(transport);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
PumpMessage msg = RoundtripService.getInstance().pumpManager.getBolusWizardCarbProfile();
|
||||
ServiceResult result = getServiceTransport().getServiceResult();
|
||||
// interpret msg here.
|
||||
getServiceTransport().setServiceResult(result);
|
||||
}
|
||||
}
|
|
@ -1,37 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import android.os.Bundle;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.ISFTable;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.TimeFormat;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/10/16.
|
||||
*/
|
||||
public class ReadISFProfileTask extends PumpTask {
|
||||
public ReadISFProfileTask() {}
|
||||
public ReadISFProfileTask(ServiceTransport transport) { super(transport); }
|
||||
@Override
|
||||
public void preOp() {
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
ISFTable table = RoundtripService.getInstance().pumpManager.getPumpISFProfile();
|
||||
ServiceResult result = getServiceTransport().getServiceResult();
|
||||
if (table.isValid()) {
|
||||
// convert from ISFTable to ISFProfile
|
||||
Bundle map = result.getMap();
|
||||
map.putIntArray("times", table.getTimes());
|
||||
map.putFloatArray("rates", table.getRates());
|
||||
map.putString("ValidDate", TimeFormat.standardFormatter().print(table.getValidDate()));
|
||||
result.setMap(map);
|
||||
result.setResultOK();
|
||||
getServiceTransport().setServiceResult(result);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -1,29 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ReadPumpClockResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/9/16.
|
||||
*/
|
||||
public class ReadPumpClockTask extends PumpTask {
|
||||
private static final String TAG = "ReadPumpClockTask";
|
||||
public ReadPumpClockTask() { }
|
||||
public ReadPumpClockTask(ServiceTransport transport) {
|
||||
super(transport);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
ReadPumpClockResult pumpResponse = RoundtripService.getInstance().pumpManager.getPumpRTC();
|
||||
if (pumpResponse != null) {
|
||||
Log.i(TAG, "ReadPumpClock: " + pumpResponse.getTimeString());
|
||||
} else {
|
||||
Log.e(TAG, "handleServiceCommand(" + mTransport.getOriginalCommandName() + ") pumpResponse is null");
|
||||
}
|
||||
getServiceTransport().setServiceResult(pumpResponse);
|
||||
}
|
||||
}
|
|
@ -1,37 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.Page;
|
||||
import com.gxwtech.roundtrip2.ServiceData.RetrieveHistoryPageResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/9/16.
|
||||
*/
|
||||
public class RetrieveHistoryPageTask extends PumpTask {
|
||||
public RetrieveHistoryPageTask() { }
|
||||
public RetrieveHistoryPageTask(ServiceTransport transport) {
|
||||
super(transport);
|
||||
}
|
||||
private Page page;
|
||||
private RetrieveHistoryPageResult result;
|
||||
private int pageNumber;
|
||||
|
||||
@Override
|
||||
public void preOp() {
|
||||
// This is to avoid allocating any memory from async thread, though I'm not sure it's necessary.
|
||||
RetrieveHistoryPageResult result = new RetrieveHistoryPageResult();
|
||||
getServiceTransport().setServiceResult(result);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
pageNumber = mTransport.getServiceCommand().getMap().getInt("pageNumber");
|
||||
page = RoundtripService.getInstance().pumpManager.getPumpHistoryPage(pageNumber);
|
||||
result = (RetrieveHistoryPageResult) getServiceTransport().getServiceResult();
|
||||
result.setResultOK();
|
||||
result.setPageNumber(pageNumber);
|
||||
result.setPageBundle(page.pack());
|
||||
}
|
||||
|
||||
}
|
|
@ -1,50 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import android.os.AsyncTask;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/9/16.
|
||||
*/
|
||||
public class ServiceTask implements Runnable {
|
||||
private static final String TAG = "ServiceTask(base)";
|
||||
protected ServiceTransport mTransport;
|
||||
public boolean completed = false;
|
||||
public ServiceTask() {
|
||||
init(new ServiceTransport());
|
||||
}
|
||||
public ServiceTask(ServiceTransport transport) {
|
||||
init(transport);
|
||||
}
|
||||
|
||||
public void init(ServiceTransport transport) {
|
||||
mTransport = transport;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
}
|
||||
|
||||
public void preOp() {
|
||||
// This function is called by UI thread before running asynch thread.
|
||||
}
|
||||
|
||||
public void postOp() {
|
||||
// This function is called by UI thread after running asynch thread.
|
||||
}
|
||||
|
||||
public ServiceTransport getServiceTransport() {
|
||||
return mTransport;
|
||||
}
|
||||
|
||||
/*
|
||||
protected void sendResponse(ServiceResult result) {
|
||||
RoundtripService.getInstance().sendServiceTransportResponse(mTransport,result);
|
||||
}
|
||||
*/
|
||||
}
|
||||
|
|
@ -1,42 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
|
||||
import java.util.concurrent.LinkedBlockingQueue;
|
||||
import java.util.concurrent.ThreadPoolExecutor;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/9/16.
|
||||
*/
|
||||
public class ServiceTaskExecutor extends ThreadPoolExecutor {
|
||||
private static final String TAG = "ServiceTaskExecutor";
|
||||
private static ServiceTaskExecutor instance;
|
||||
private static LinkedBlockingQueue<Runnable> taskQueue = new LinkedBlockingQueue<>();
|
||||
static {
|
||||
instance = new ServiceTaskExecutor();
|
||||
}
|
||||
private ServiceTaskExecutor() {
|
||||
super(1,1,10000, TimeUnit.MILLISECONDS,taskQueue);
|
||||
}
|
||||
public static ServiceTask startTask(ServiceTask task) {
|
||||
instance.execute(task); // task will be run on async thread from pool.
|
||||
return task;
|
||||
}
|
||||
protected void beforeExecute(Thread t, Runnable r) {
|
||||
// This is run on either caller UI thread or Service UI thread.
|
||||
ServiceTask task = (ServiceTask) r;
|
||||
Log.v(TAG,"About to run task " + task.getClass().getSimpleName());
|
||||
RoundtripService.getInstance().setCurrentTask(task);
|
||||
task.preOp();
|
||||
}
|
||||
protected void afterExecute(Runnable r, Throwable t) {
|
||||
// This is run on either caller UI thread or Service UI thread.
|
||||
ServiceTask task = (ServiceTask) r;
|
||||
task.postOp();
|
||||
Log.v(TAG,"Finishing task " + task.getClass().getSimpleName());
|
||||
RoundtripService.getInstance().finishCurrentTask(task);
|
||||
}
|
||||
}
|
|
@ -1,58 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RileyLink.PumpManagerStatus;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.ServiceData.PumpStatusResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceResult;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/16/16.
|
||||
*/
|
||||
public class UpdatePumpStatusTask extends PumpTask {
|
||||
private static final String TAG = "UpdatePumpStatusTask";
|
||||
public UpdatePumpStatusTask() { }
|
||||
public UpdatePumpStatusTask(ServiceTransport transport) {
|
||||
super(transport);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
// force pump to update everything it can
|
||||
RoundtripService.getInstance().pumpManager.updatePumpManagerStatus();
|
||||
// get the newly cached status
|
||||
PumpManagerStatus status = RoundtripService.getInstance().pumpManager.getPumpManagerStatus();
|
||||
// fill a PumpStatusResult message with the goods
|
||||
PumpStatusResult result = new PumpStatusResult();
|
||||
/*
|
||||
double remainBattery;
|
||||
double remainUnits;
|
||||
double currentBasal;
|
||||
double lastBolusAmount;
|
||||
String lastBolusTime;
|
||||
int tempBasalInProgress;
|
||||
double tempBasalRatio;
|
||||
double tempBasalRemainMin;
|
||||
String tempBasalStart;
|
||||
String time;
|
||||
String timeLastSync;
|
||||
*/
|
||||
|
||||
result.setRemainBattery(status.remainBattery);
|
||||
result.setRemainUnits(status.remainUnits);
|
||||
result.setCurrentBasal(status.currentBasal);
|
||||
result.setLastBolusAmount(status.last_bolus_amount);
|
||||
result.setLastBolusTime(status.last_bolus_time.toString());
|
||||
result.setTempBasalInProgress(status.tempBasalInProgress);
|
||||
result.setTempBasalRatio(status.tempBasalRatio);
|
||||
result.setTempBasalRemainMin(status.tempBasalRemainMin);
|
||||
result.setTempBasalStart(status.tempBasalStart.toString());
|
||||
result.setTime(status.time.toString());
|
||||
result.setTimeLastSync("");
|
||||
getServiceTransport().setServiceResult(result);
|
||||
|
||||
}
|
||||
|
||||
}
|
|
@ -1,22 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.Tasks;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.RoundtripService;
|
||||
import com.gxwtech.roundtrip2.ServiceData.ServiceTransport;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/16/16.
|
||||
*/
|
||||
public class WakeAndTuneTask extends PumpTask {
|
||||
private static final String TAG = "WakeAndTuneTask";
|
||||
public WakeAndTuneTask() { }
|
||||
public WakeAndTuneTask(ServiceTransport transport) {
|
||||
super(transport);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
RoundtripService.getInstance().pumpManager.wakeup(6);
|
||||
RoundtripService.getInstance().pumpManager.tuneForPump();
|
||||
}
|
||||
|
||||
}
|
|
@ -1,23 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/2/16.
|
||||
*/
|
||||
public class ButtonPressCarelinkMessageBody extends CarelinkLongMessageBody {
|
||||
public static final byte BUTTON_EASY = 0x00;
|
||||
public static final byte BUTTON_ESC = 0x01;
|
||||
public static final byte BUTTON_ACT = 0x02;
|
||||
public static final byte BUTTON_UP = 0x03;
|
||||
public static final byte BUTTON_DOWN = 0x04;
|
||||
|
||||
public ButtonPressCarelinkMessageBody(int which) {
|
||||
init(which);
|
||||
}
|
||||
|
||||
public void init(int buttonType) {
|
||||
int numArgs = 1;
|
||||
super.init(new byte[] {(byte)numArgs,(byte)buttonType});
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,36 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/2/16.
|
||||
*/
|
||||
public class CarelinkLongMessageBody extends MessageBody {
|
||||
public static final int LONG_MESSAGE_BODY_LENGTH = 65;
|
||||
protected byte[] data;
|
||||
|
||||
public CarelinkLongMessageBody() {
|
||||
init(new byte[0]);
|
||||
}
|
||||
|
||||
public CarelinkLongMessageBody(byte[] payload) {
|
||||
init(payload);
|
||||
}
|
||||
|
||||
public void init(byte[] rxData) {
|
||||
data = new byte[LONG_MESSAGE_BODY_LENGTH];
|
||||
if (rxData != null) {
|
||||
int size = rxData.length < LONG_MESSAGE_BODY_LENGTH ? rxData.length : LONG_MESSAGE_BODY_LENGTH;
|
||||
for (int i=0; i<size; i++) {
|
||||
data[i] = rxData[i];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public int getLength() {
|
||||
return LONG_MESSAGE_BODY_LENGTH;
|
||||
}
|
||||
|
||||
public byte[] getTxData() {
|
||||
return data;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,38 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/16.
|
||||
*/
|
||||
public class CarelinkShortMessageBody extends MessageBody {
|
||||
byte[] body;
|
||||
|
||||
public int getLength() {
|
||||
return body.length;
|
||||
}
|
||||
|
||||
public CarelinkShortMessageBody() { init(new byte[] {0}); }
|
||||
|
||||
public CarelinkShortMessageBody(byte[] data) {
|
||||
init(data);
|
||||
}
|
||||
|
||||
public void init(byte[] rxData) {
|
||||
body = rxData;
|
||||
}
|
||||
|
||||
public byte[] getRxData() {
|
||||
return body;
|
||||
}
|
||||
|
||||
public void setRxData(byte[] rxData) {
|
||||
init(rxData);
|
||||
}
|
||||
|
||||
public byte[] getTxData() {
|
||||
return body;
|
||||
}
|
||||
|
||||
public void setTxData(byte[] txData) {
|
||||
init(txData);
|
||||
}
|
||||
}
|
|
@ -1,50 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/2/16.
|
||||
*/
|
||||
public class GetHistoryPageCarelinkMessageBody extends CarelinkLongMessageBody {
|
||||
//public boolean wasLastFrame = false;
|
||||
//public int frameNumber = 0;
|
||||
//public byte[] frame = new byte[] {};
|
||||
|
||||
public GetHistoryPageCarelinkMessageBody(byte[] frameData) {
|
||||
init(frameData);
|
||||
}
|
||||
|
||||
public GetHistoryPageCarelinkMessageBody(int pageNum) {
|
||||
init(pageNum);
|
||||
}
|
||||
|
||||
public int getLength() {
|
||||
return data.length;
|
||||
}
|
||||
|
||||
public void init(byte[] rxData) {
|
||||
super.init(rxData);
|
||||
}
|
||||
|
||||
public void init(int pageNum) {
|
||||
byte numArgs = 1;
|
||||
super.init(new byte[] {numArgs,(byte)pageNum});
|
||||
}
|
||||
|
||||
public int getFrameNumber() {
|
||||
if (data.length > 0) {
|
||||
return data[0] & 0x7f;
|
||||
}
|
||||
return 255;
|
||||
}
|
||||
|
||||
public boolean wasLastFrame() {
|
||||
return (data[0] & 0x80) != 0;
|
||||
}
|
||||
|
||||
public byte[] getFrameData() {
|
||||
return ByteUtil.substring(data,1,data.length-1);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,31 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/16.
|
||||
*/
|
||||
public class GetPumpModelCarelinkMessageBody extends MessageBody {
|
||||
|
||||
public int getLength() {
|
||||
return 1;
|
||||
}
|
||||
|
||||
public void init(byte[] rxData) {
|
||||
|
||||
}
|
||||
|
||||
public byte[] getRxData() {
|
||||
return new byte[] { 0 };
|
||||
}
|
||||
|
||||
public void setRxData(byte[] rxData) {
|
||||
|
||||
}
|
||||
|
||||
public byte[] getTxData() {
|
||||
return new byte[] { 0 };
|
||||
}
|
||||
|
||||
public void setTxData(byte[] txData) {
|
||||
|
||||
}
|
||||
}
|
|
@ -1,10 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/16.
|
||||
*/
|
||||
public class MessageBody {
|
||||
public int getLength() { return 0; }
|
||||
public void init(byte[] rxData) { }
|
||||
public byte[] getTxData() { return new byte[] {}; }
|
||||
}
|
|
@ -1,105 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/16.
|
||||
*/
|
||||
public class MessageType {
|
||||
public static final byte Invalid = (byte)0x00;
|
||||
public static final byte Alert = (byte)0x01;
|
||||
public static final byte AlertCleared = (byte)0x02;
|
||||
public static final byte DeviceTest = (byte)0x03;
|
||||
public static final byte PumpStatus = (byte)0x04;
|
||||
public static final byte PumpAck = (byte)0x06;
|
||||
public static final byte PumpBackfill = (byte)0x08;
|
||||
public static final byte FindDevice = (byte)0x09;
|
||||
public static final byte DeviceLink = (byte)0x0a;
|
||||
public static final byte ChangeTime = (byte)0x40;
|
||||
public static final byte Bolus = (byte)0x42;
|
||||
public static final byte ChangeTempBasal = (byte)0x4c;
|
||||
public static final byte ButtonPress = (byte)0x5b;
|
||||
public static final byte PowerOn = (byte)0x5d;
|
||||
public static final byte ReadTime = (byte)0x70;
|
||||
public static final byte GetBattery = (byte)0x72;
|
||||
public static final byte GetHistoryPage = (byte)0x80;
|
||||
public static final byte GetISFProfile = (byte)0x8b;
|
||||
public static final byte GetPumpModel = (byte)0x8d;
|
||||
public static final byte ReadTempBasal = (byte)0x98;
|
||||
public static final byte ReadSettings = (byte)0xc0;
|
||||
|
||||
// The above codes include codes that are not 522/722 specific.
|
||||
|
||||
// The codes below here are Medtronic pump specific.
|
||||
// from Roundtrip.Carelink:
|
||||
public static final byte CMD_M_PACKET_LENGTH = ((byte)7); // 0x07
|
||||
public static final byte CMD_M_BEGIN_PARAMETER_SETTING = ((byte)38); // 0x26
|
||||
public static final byte CMD_M_END_PARAMETER_SETTING = ((byte)39); // 0x27
|
||||
public static final byte CMD_M_SET_A_PROFILE = ((byte)48); // 0x30
|
||||
public static final byte CMD_M_SET_B_PROFILE = ((byte)49); // 0x31
|
||||
public static final byte CMD_M_SET_LOGIC_LINK_ID = ((byte)50); // 0x32
|
||||
public static final byte CMD_M_SET_LOGIC_LINK_ENABLE = ((byte)51); // 0x33
|
||||
public static final byte CMD_M_SET_RTC = ((byte)64); // 0x40
|
||||
public static final byte CMD_M_SET_MAX_BOLUS = ((byte)65); // 0x41
|
||||
public static final byte CMD_M_BOLUS = ((byte)66); // 0x42
|
||||
public static final byte CMD_M_SET_VAR_BOLUS_ENABLE = ((byte)69); // 0x45
|
||||
public static final byte CMD_M_SET_CURRENT_PATTERN = ((byte)74); // 0x4a
|
||||
public static final byte CMD_M_TEMP_BASAL_RATE = ((byte)76); // 0x4c
|
||||
public static final byte CMD_M_SUSPEND_RESUME = ((byte)77); // 0x4d
|
||||
public static final byte CMD_M_SET_AUTO_OFF = ((byte)78); // 0x4e
|
||||
public static final byte CMD_M_SET_EASY_BOLUS_ENABLE = ((byte)79); // 0x4f
|
||||
public static final byte CMD_M_SET_RF_REMOTE_ID = ((byte)81); // 0x51
|
||||
public static final byte CMD_M_SET_BLOCK_ENABLE = ((byte)82); // 0x52
|
||||
public static final byte CMD_M_SET_ALERT_TYPE = ((byte)84); // 0x54
|
||||
public static final byte CMD_M_SET_PATTERNS_ENABLE = ((byte)85); // 0x55
|
||||
public static final byte CMD_M_SET_RF_ENABLE = ((byte)87); // 0x57
|
||||
public static final byte CMD_M_SET_INSULIN_ACTION_TYPE = ((byte)88); // 0x58
|
||||
public static final byte CMD_M_KEYPAD_PUSH = ((byte)91); // 0x5b
|
||||
public static final byte CMD_M_SET_TIME_FORMAT = ((byte)92); // 0x5c
|
||||
public static final byte CMD_M_POWER_CTRL = ((byte)93); // 0x5d
|
||||
public static final byte CMD_M_SET_BOLUS_WIZARD_SETUP = ((byte)94); // 0x5e
|
||||
public static final byte CMD_M_SET_BG_ALARM_ENABLE = ((byte)103);// 0x67
|
||||
public static final byte CMD_M_SET_TEMP_BASAL_TYPE = ((byte)104);// 0x68
|
||||
public static final byte CMD_M_SET_RESERVOIR_WARNING = ((byte)106);// 0x6a
|
||||
public static final byte CMD_M_SET_BG_ALARM_CLOCKS = ((byte)107);// 0x6b
|
||||
public static final byte CMD_M_SET_BG_REMINDER_ENABLE = ((byte)108);// 0x6c
|
||||
public static final byte CMD_M_SET_MAX_BASAL = ((byte)110);// 0x6e
|
||||
public static final byte CMD_M_SET_STD_PROFILE = ((byte)111);// 0x6f
|
||||
public static final byte CMD_M_READ_RTC = ((byte)112);// 0x70
|
||||
public static final byte CMD_M_READ_PUMP_ID = ((byte)113);// 0x71
|
||||
public static final byte CMD_M_READ_INSULIN_REMAINING = ((byte)115);// 0x73
|
||||
public static final byte CMD_M_READ_FIRMWARE_VER = ((byte)116);// 0x74
|
||||
public static final byte CMD_M_READ_ERROR_STATUS = ((byte)117);// 0x75
|
||||
public static final byte CMD_M_READ_REMOTE_CTRL_IDS = ((byte)118);// 0x76
|
||||
public static final byte CMD_M_READ_HISTORY = ((byte)128);// 0x80
|
||||
public static final byte CMD_M_READ_PUMP_STATE = ((byte)131);// 0x83
|
||||
public static final byte CMD_M_READ_BOLUS_WIZARD_SETUP_STATUS = ((byte)135);// 0x87
|
||||
public static final byte CMD_M_READ_CARB_UNITS = ((byte)136);// 0x88
|
||||
public static final byte CMD_M_READ_BG_UNITS = ((byte)137);// 0x89
|
||||
public static final byte CMD_M_READ_CARB_RATIOS = ((byte)138);// 0x8a
|
||||
public static final byte CMD_M_READ_INSULIN_SENSITIVITIES = ((byte)139);// 0x8b
|
||||
public static final byte CMD_M_READ_BG_TARGETS = ((byte)140);// 0x8c
|
||||
public static final byte CMD_M_READ_PUMP_MODEL_NUMBER = ((byte)141);// 0x8d
|
||||
public static final byte CMD_M_READ_BG_ALARM_CLOCKS = ((byte)142);// 0x8e
|
||||
public static final byte CMD_M_READ_RESERVOIR_WARNING = ((byte)143);// 0x8f
|
||||
public static final byte CMD_M_READ_BG_REMINDER_ENABLE = ((byte)144);// 0x90
|
||||
public static final byte CMD_M_READ_SETTINGS = ((byte)145);// 0x91
|
||||
public static final byte CMD_M_READ_STD_PROFILES = ((byte)146);// 0x92
|
||||
public static final byte CMD_M_READ_A_PROFILES = ((byte)147);// 0x93
|
||||
public static final byte CMD_M_READ_B_PROFILES = ((byte)148);// 0x94
|
||||
public static final byte CMD_M_READ_LOGIC_LINK_IDS = ((byte)149);// 0x95
|
||||
public static final byte CMD_M_READ_BG_ALARM_ENABLE = ((byte)151);// 0x97
|
||||
public static final byte CMD_M_READ_TEMP_BASAL = ((byte)152);// 0x98
|
||||
public static final byte CMD_M_READ_PUMP_SETTINGS = ((byte)192);// 0xc0
|
||||
public static final byte CMD_M_READ_PUMP_STATUS = ((byte)206);// 0xce
|
||||
|
||||
public byte mtype;
|
||||
public MessageType(byte mtype) {
|
||||
this.mtype = mtype;
|
||||
}
|
||||
|
||||
public static MessageBody constructMessageBody(MessageType messageType, byte[] bodyData) {
|
||||
switch (messageType.mtype) {
|
||||
case PumpAck: return new PumpAckMessageBody(bodyData);
|
||||
default: return new UnknownMessageBody(bodyData);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,11 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/16.
|
||||
*/
|
||||
public class PumpAckMessageBody extends CarelinkShortMessageBody {
|
||||
public PumpAckMessageBody() { init(new byte[] {0}); }
|
||||
public PumpAckMessageBody(byte[] bodyData) {
|
||||
init(bodyData);
|
||||
}
|
||||
}
|
|
@ -1,35 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.Messages;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/16.
|
||||
*/
|
||||
public class UnknownMessageBody extends MessageBody {
|
||||
public byte[] rxData;
|
||||
|
||||
public int getLength() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
public UnknownMessageBody(byte[] data) {
|
||||
this.rxData = data;
|
||||
}
|
||||
|
||||
public void init(byte[] rxData) {
|
||||
}
|
||||
|
||||
public byte[] getRxData() {
|
||||
return rxData;
|
||||
}
|
||||
|
||||
public void setRxData(byte[] rxData) {
|
||||
this.rxData = rxData;
|
||||
}
|
||||
|
||||
public byte[] getTxData() {
|
||||
return rxData;
|
||||
}
|
||||
|
||||
public void setTxData(byte[] txData) {
|
||||
this.rxData = txData;
|
||||
}
|
||||
}
|
|
@ -1,20 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/16.
|
||||
*/
|
||||
public class PacketType {
|
||||
public static final short Invalid = 0x00;
|
||||
public static final short MySentry = 0xa2;
|
||||
public static final short Meter = 0xa5;
|
||||
public static final short Carelink = 0xa7;
|
||||
public static final short Sensor = 0xa8;
|
||||
|
||||
public short value = 0;
|
||||
public PacketType() {
|
||||
}
|
||||
public PacketType(short value) {
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,175 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import android.util.Log;
|
||||
|
||||
import org.joda.time.Instant;
|
||||
|
||||
import java.util.ArrayList;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/1/15.
|
||||
*
|
||||
* There are three basal profiles stored on the pump. (722 only?)
|
||||
* They are all parsed the same, the user just has 3 to choose from:
|
||||
* Standard, A, and B
|
||||
*
|
||||
* The byte array seems to be 21 three byte entries long, plus a zero?
|
||||
* If the profile is completely empty, it should have one entry: [0,0,0x3F] (?)
|
||||
* The first entry of [0,0,0] marks the end of the used entries.
|
||||
*
|
||||
* Each entry is assumed to span from the specified start time to the start time of the
|
||||
* next entry, or to midnight if there are no more entries.
|
||||
*
|
||||
* Individual entries are of the form [r,z,m] where
|
||||
* r is the rate (in 0.025 U increments)
|
||||
* z is zero (?)
|
||||
* m is the start time-of-day for the basal rate period (in 30 minute increments?)
|
||||
*/
|
||||
public class BasalProfile {
|
||||
private static final String TAG = "BasalProfile";
|
||||
private static final boolean DEBUG_BASALPROFILE = false;
|
||||
protected static final int MAX_RAW_DATA_SIZE = (21 * 3) + 1;
|
||||
protected byte[] mRawData; // store as byte array to make transport (via parcel) easier
|
||||
public BasalProfile() {
|
||||
init();
|
||||
}
|
||||
public void init() {
|
||||
mRawData = new byte[MAX_RAW_DATA_SIZE];
|
||||
mRawData[0] = 0;
|
||||
mRawData[1] = 0;
|
||||
mRawData[2] = 0x3f;
|
||||
}
|
||||
// this asUINT8 should be combined with Record.asUINT8, and placed in a new util class.
|
||||
protected static int readUnsignedByte(byte b) { return (b<0)?b+256:b; }
|
||||
public boolean setRawData(byte[] data) {
|
||||
if (data == null) {
|
||||
Log.e(TAG,"setRawData: buffer is null!");
|
||||
return false;
|
||||
}
|
||||
int len = Math.min(MAX_RAW_DATA_SIZE, data.length);
|
||||
System.arraycopy(data, 0, mRawData, 0, len);
|
||||
if (DEBUG_BASALPROFILE) {
|
||||
Log.v(TAG, String.format("setRawData: copied raw data buffer of %d bytes.", len));
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public void dumpBasalProfile() {
|
||||
Log.v(TAG, "Basal Profile entries:");
|
||||
ArrayList<BasalProfileEntry> entries = getEntries();
|
||||
for (int i=0; i< entries.size(); i++) {
|
||||
BasalProfileEntry entry = entries.get(i);
|
||||
String startString = entry.startTime.toString("HH:mm");
|
||||
Log.v(TAG, String.format("Entry %d, rate=%.3f (0x%02X), start=%s (0x%02X)",
|
||||
i+1, entry.rate, entry.rate_raw,
|
||||
startString, entry.startTime_raw));
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: this function must be expanded to include changes in which profile is in use.
|
||||
// and changes to the profiles themselves.
|
||||
public BasalProfileEntry getEntryForTime(Instant when) {
|
||||
BasalProfileEntry rval = new BasalProfileEntry();
|
||||
ArrayList<BasalProfileEntry> entries = getEntries();
|
||||
if (entries.size() == 0) {
|
||||
Log.w(TAG, String.format("getEntryForTime(%s): table is empty",
|
||||
when.toDateTime().toLocalTime().toString("HH:mm")));
|
||||
return rval;
|
||||
}
|
||||
//Log.w(TAG,"Assuming first entry");
|
||||
rval = entries.get(0);
|
||||
if (entries.size() == 1) {
|
||||
Log.v(TAG,"getEntryForTime: Only one entry in profile");
|
||||
return rval;
|
||||
}
|
||||
|
||||
int localMillis = when.toDateTime().toLocalTime().getMillisOfDay();
|
||||
boolean done = false;
|
||||
int i=1;
|
||||
while (!done) {
|
||||
BasalProfileEntry entry = entries.get(i);
|
||||
if (DEBUG_BASALPROFILE) {
|
||||
Log.v(TAG, String.format("Comparing 'now'=%s to entry 'start time'=%s",
|
||||
when.toDateTime().toLocalTime().toString("HH:mm"),
|
||||
entry.startTime.toString("HH:mm")));
|
||||
}
|
||||
if (localMillis >= entry.startTime.getMillisOfDay()) {
|
||||
rval = entry;
|
||||
if (DEBUG_BASALPROFILE) Log.v(TAG,"Accepted Entry");
|
||||
} else {
|
||||
// entry at i has later start time, keep older entry
|
||||
if (DEBUG_BASALPROFILE) Log.v(TAG,"Rejected Entry");
|
||||
done = true;
|
||||
}
|
||||
i++;
|
||||
if (i >= entries.size()) {
|
||||
done = true;
|
||||
}
|
||||
}
|
||||
if (DEBUG_BASALPROFILE) {
|
||||
Log.v(TAG, String.format("getEntryForTime(%s): Returning entry: rate=%.3f (%d), start=%s (%d)",
|
||||
when.toDateTime().toLocalTime().toString("HH:mm"),
|
||||
rval.rate, rval.rate_raw,
|
||||
rval.startTime.toString("HH:mm"), rval.startTime_raw));
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
public ArrayList<BasalProfileEntry> getEntries() {
|
||||
ArrayList<BasalProfileEntry> entries = new ArrayList<>();
|
||||
if (mRawData[2] == 0x3f) {
|
||||
Log.w(TAG,"Raw Data is empty.");
|
||||
return entries; // an empty list
|
||||
}
|
||||
int i = 0;
|
||||
boolean done = false;
|
||||
int r,st;
|
||||
while (!done) {
|
||||
r = readUnsignedByte(mRawData[i]);
|
||||
// What is mRawData[i+1]? Not used in decocare.
|
||||
st = readUnsignedByte(mRawData[i+2]);
|
||||
entries.add(new BasalProfileEntry(r,st));
|
||||
i=i+3;
|
||||
if (i>=MAX_RAW_DATA_SIZE) {
|
||||
done=true;
|
||||
} else if ((mRawData[i]==0) && (mRawData[i+1]==0) && (mRawData[i+2]==0)) {
|
||||
done = true;
|
||||
}
|
||||
}
|
||||
return entries;
|
||||
}
|
||||
|
||||
public static void testParser() {
|
||||
byte[] testData = new byte[] {
|
||||
32, 0, 0,
|
||||
38, 0, 13,
|
||||
44, 0, 19,
|
||||
38, 0, 28,
|
||||
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||
0, 0, 0, 0, 0, 0, 0, 0 };
|
||||
/* from decocare:
|
||||
_test_schedule = {'total': 22.50, 'schedule': [
|
||||
{ 'start': '12:00A', 'rate': 0.80 },
|
||||
{ 'start': '6:30A', 'rate': 0.95 },
|
||||
{ 'start': '9:30A', 'rate': 1.10 },
|
||||
{ 'start': '2:00P', 'rate': 0.95 },
|
||||
]}
|
||||
*/
|
||||
BasalProfile profile = new BasalProfile();
|
||||
profile.setRawData(testData);
|
||||
ArrayList<BasalProfileEntry> entries = profile.getEntries();
|
||||
if (entries.isEmpty()) {
|
||||
Log.e(TAG,"testParser: failed");
|
||||
} else {
|
||||
for (int i=0; i<entries.size(); i++) {
|
||||
BasalProfileEntry e = entries.get(i);
|
||||
Log.d(TAG, String.format("testParser entry #%d: rate: %.2f, start %d:%d",
|
||||
i,e.rate,e.startTime.getHourOfDay(),
|
||||
e.startTime.getMinuteOfHour()));
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
|
@ -1,28 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import org.joda.time.LocalTime;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/1/15.
|
||||
* This is a helper class for BasalProfile, only used for interpreting the contents of BasalProfile
|
||||
*/
|
||||
public class BasalProfileEntry {
|
||||
public byte rate_raw;
|
||||
public double rate;
|
||||
public byte startTime_raw;
|
||||
public LocalTime startTime; // Just a "time of day"
|
||||
public BasalProfileEntry() {
|
||||
rate = -9.999E6;
|
||||
rate_raw = (byte)0xFF;
|
||||
startTime = new LocalTime(0);
|
||||
startTime_raw = (byte)0xFF;
|
||||
}
|
||||
public BasalProfileEntry(int rateByte, int startTimeByte) {
|
||||
// rateByte is insulin delivery rate, U/hr, in 0.025 U increments
|
||||
// startTimeByte is time-of-day, in 30 minute increments
|
||||
rate_raw = (byte)rateByte;
|
||||
rate = rateByte * 0.025;
|
||||
startTime_raw = (byte)startTimeByte;
|
||||
startTime = new LocalTime(startTimeByte / 2, (startTimeByte % 2) * 30);
|
||||
}
|
||||
}
|
|
@ -1,8 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/1/15.
|
||||
*/
|
||||
public enum BasalProfileTypeEnum {
|
||||
STD,A,B;
|
||||
}
|
|
@ -1,31 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.BolusWizardBolusEstimatePumpEvent;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.TempBasalEvent;
|
||||
|
||||
import java.util.ArrayList;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/15.
|
||||
*
|
||||
* This class is inteded to gather what information we've gleaned from the pump history
|
||||
* into one place, make it easier to move around.
|
||||
*
|
||||
*/
|
||||
@Deprecated
|
||||
public class HistoryReport {
|
||||
public ArrayList<BolusWizardBolusEstimatePumpEvent> mBolusWizardEvents;
|
||||
public ArrayList<TempBasalEvent> mBasalEvents;
|
||||
public HistoryReport() {
|
||||
mBolusWizardEvents = new ArrayList<>();
|
||||
mBasalEvents = new ArrayList<>();
|
||||
}
|
||||
public void addBolusWizardEvent(BolusWizardBolusEstimatePumpEvent event) {
|
||||
mBolusWizardEvents.add(event);
|
||||
}
|
||||
public void addTempBasalEvent(TempBasalEvent event)
|
||||
{
|
||||
mBasalEvents.add(event);
|
||||
}
|
||||
}
|
|
@ -1,11 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/24/16.
|
||||
*/
|
||||
public class HtmlByte extends HtmlElement {
|
||||
private byte data = 0;
|
||||
public HtmlByte() {}
|
||||
public HtmlByte(byte b) { data = b; }
|
||||
public String toString() {return String.format("%02x",data);}
|
||||
}
|
|
@ -1,9 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/24/16.
|
||||
*/
|
||||
public class HtmlCodeTagEnd extends HtmlElement {
|
||||
public HtmlCodeTagEnd() {}
|
||||
public String toString() { return "</code>"; }
|
||||
}
|
|
@ -1,30 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/24/16.
|
||||
*/
|
||||
public class HtmlCodeTagStart extends HtmlElement {
|
||||
String title = "";
|
||||
String color = "";
|
||||
public HtmlCodeTagStart(String title, String bgColor) {
|
||||
this.title = title;
|
||||
this.color = bgColor;
|
||||
}
|
||||
public String toString() {
|
||||
StringBuilder b = new StringBuilder();
|
||||
b.append("<code");
|
||||
|
||||
if ((title != null) && (title.length() > 0)) {
|
||||
b.append(" title=\"");
|
||||
b.append(title);
|
||||
b.append("\"");
|
||||
}
|
||||
if ((color != null) && (color.length() > 0)) {
|
||||
b.append(" style=\"background-color:");
|
||||
b.append(color);
|
||||
b.append(";\"");
|
||||
}
|
||||
b.append(">");
|
||||
return b.toString();
|
||||
}
|
||||
}
|
|
@ -1,9 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/24/16.
|
||||
*/
|
||||
abstract public class HtmlElement {
|
||||
public HtmlElement() {}
|
||||
abstract public String toString();
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/24/16.
|
||||
*/
|
||||
public class HtmlElementGeneric extends HtmlElement {
|
||||
String s = "";
|
||||
public HtmlElementGeneric() {}
|
||||
public HtmlElementGeneric(String s) {
|
||||
this.s = s;
|
||||
}
|
||||
public String toString() {
|
||||
return s;
|
||||
}
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/24/16.
|
||||
*/
|
||||
public class HtmlHistoryPageStart extends HtmlElement {
|
||||
int pagenum = 0;
|
||||
public HtmlHistoryPageStart() {}
|
||||
public HtmlHistoryPageStart(int num) {
|
||||
pagenum = num;
|
||||
}
|
||||
public String toString() {
|
||||
return "<h2>History Page " + pagenum + "</h2>\n";
|
||||
}
|
||||
}
|
|
@ -1,102 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
|
||||
import org.joda.time.LocalDateTime;
|
||||
|
||||
/**
|
||||
* Created by geoff on 7/2/16.
|
||||
*
|
||||
* This class exists to map the Minimed byte response to "Get sensitivity factors"
|
||||
* into a more usable form.
|
||||
*/
|
||||
public class ISFTable {
|
||||
boolean mIsValid = false;
|
||||
|
||||
LocalDateTime validDate;
|
||||
byte[] header;
|
||||
int[] times;
|
||||
float[] rates;
|
||||
public ISFTable() {}
|
||||
public boolean parseFrom(byte[] responseBytes) {
|
||||
// example format: { 7, 1, 0, 45, 12, 30, 42, 50, 0, 0 }
|
||||
// means value pairs: {0, 45}, {12, 30}, {42, 50}
|
||||
// means: at midnight, the amount is 45,
|
||||
// at 6am, the about is 30
|
||||
// at 9pm, the amount is 50
|
||||
if (responseBytes == null) return false;
|
||||
// minimum of two bytes in header
|
||||
if (responseBytes.length < 2) {
|
||||
return false;
|
||||
}
|
||||
// Must be an even number of times and rates
|
||||
if (responseBytes.length % 2 != 0) {
|
||||
return false;
|
||||
}
|
||||
mIsValid = true;
|
||||
header = ByteUtil.substring(responseBytes,0,2);
|
||||
// find end of list
|
||||
int index = 2;
|
||||
while (true) {
|
||||
if (index + 1 > responseBytes.length) {
|
||||
break;
|
||||
}
|
||||
if ((responseBytes[index]==0) && (responseBytes[index+1]==0)) {
|
||||
break;
|
||||
}
|
||||
index += 2;
|
||||
}
|
||||
if (index == 2) {
|
||||
// no entries.
|
||||
times = new int[]{};
|
||||
rates = new float[]{};
|
||||
} else {
|
||||
int numEntries = (index - 2) / 2;
|
||||
times = new int[numEntries];
|
||||
rates = new float[numEntries];
|
||||
for (int i=0; i<numEntries; i++) {
|
||||
times[i] = responseBytes[i*2+2];
|
||||
rates[i] = responseBytes[i*2+2];
|
||||
}
|
||||
}
|
||||
validDate = new LocalDateTime();
|
||||
return true;
|
||||
}
|
||||
|
||||
public byte[] getHeader() {
|
||||
return header;
|
||||
}
|
||||
|
||||
public void setHeader(byte[] header) {
|
||||
this.header = header;
|
||||
}
|
||||
|
||||
public int[] getTimes() {
|
||||
return times;
|
||||
}
|
||||
|
||||
public void setTimes(int[] times) {
|
||||
this.times = times;
|
||||
}
|
||||
|
||||
public float[] getRates() {
|
||||
return rates;
|
||||
}
|
||||
|
||||
public void setRates(float[] rates) {
|
||||
this.rates = rates;
|
||||
}
|
||||
|
||||
public LocalDateTime getValidDate() {
|
||||
return validDate;
|
||||
}
|
||||
|
||||
public void setValidDate(LocalDateTime validDate) {
|
||||
this.validDate = validDate;
|
||||
}
|
||||
|
||||
public boolean isValid() {
|
||||
return mIsValid;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,409 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/13/15.
|
||||
*
|
||||
* This class was taken from medtronic-android-uploader.
|
||||
* This class was written such that the constructors did all the work, which resulted
|
||||
* in annoyances such as exceptions during constructors.
|
||||
*
|
||||
* TODO: This class needs to be revisited and probably rewritten. (2016-06-12)
|
||||
*
|
||||
*
|
||||
*
|
||||
Pete Schwamb
|
||||
@ps2
|
||||
12:04
|
||||
History entries will not reorder themselves, but some events do update, like dual wave bolus entries
|
||||
It's like an append only log, and when a page is full, it rotates the page ids and starts appending to a new blank page
|
||||
Darrell Wright
|
||||
@beached
|
||||
12:05
|
||||
so the timestamp is entry creation not update
|
||||
time
|
||||
Pete Schwamb
|
||||
@ps2
|
||||
12:06
|
||||
Yes, I don't think the timestamps ever change
|
||||
|
||||
|
||||
GGW: TODO: examine src/ecc1/medtronic for better history parsing
|
||||
*/
|
||||
|
||||
import android.os.Bundle;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.BolusWizardBolusEstimatePumpEvent;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.Record;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.RecordTypeEnum;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.TempBasalDurationPumpEvent;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.TempBasalRatePumpEvent;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpTimeStamp;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.TimeFormat;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.CRC;
|
||||
import com.gxwtech.roundtrip2.util.HexDump;
|
||||
|
||||
import org.joda.time.DateTime;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
|
||||
public class Page {
|
||||
private final static String TAG = "Page";
|
||||
private static final boolean DEBUG_PAGE = true;
|
||||
|
||||
private byte[] crc;
|
||||
private byte[] data;
|
||||
protected PumpModel model;
|
||||
public List<Record> mRecordList;
|
||||
|
||||
public Page() {
|
||||
this.model = PumpModel.UNSET;
|
||||
mRecordList = new ArrayList<>();
|
||||
}
|
||||
|
||||
public byte[] getRawData() {
|
||||
if (data == null) {
|
||||
return crc;
|
||||
}
|
||||
if (crc == null) {
|
||||
return data;
|
||||
}
|
||||
return ByteUtil.concat(data,crc);
|
||||
}
|
||||
|
||||
protected PumpTimeStamp collectTimeStamp(byte[] data, int offset) {
|
||||
try {
|
||||
PumpTimeStamp timestamp = new PumpTimeStamp(TimeFormat.parse5ByteDate(data, offset));
|
||||
return timestamp;
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
public boolean parsePicky(byte[] rawPage, PumpModel model) {
|
||||
mRecordList = new ArrayList<>();
|
||||
this.model = model;
|
||||
int pageOffset = 0;
|
||||
|
||||
if ((rawPage == null) || (rawPage.length == 0)) return false;
|
||||
this.data = Arrays.copyOfRange(rawPage, 0, rawPage.length-2);
|
||||
this.crc = Arrays.copyOfRange(rawPage, rawPage.length-2, rawPage.length);
|
||||
byte[] expectedCrc = CRC.calculate16CCITT(this.data);
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, String.format("Data length: %d", data.length));
|
||||
}
|
||||
if (!Arrays.equals(crc, expectedCrc)) {
|
||||
Log.w(TAG, String.format("CRC does not match expected value. Expected: %s Was: %s", HexDump.toHexString(expectedCrc), HexDump.toHexString(crc)));
|
||||
} else {
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, "CRC OK");
|
||||
}
|
||||
}
|
||||
|
||||
Record record = null;
|
||||
while (pageOffset < data.length) {
|
||||
if (data[pageOffset] == 0) {
|
||||
if (record != null) {
|
||||
Log.i(TAG,String.format("End of page or Previous parse fail: prev opcode 0x%02x, curr offset %d, %d bytes remaining",
|
||||
record.getRecordOp(),pageOffset,data.length - pageOffset + 1));
|
||||
break;
|
||||
} else {
|
||||
Log.i(TAG, "WTF?");
|
||||
}
|
||||
}
|
||||
try {
|
||||
record = attemptParseRecord(data, pageOffset);
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
record = null;
|
||||
}
|
||||
if (record == null) {
|
||||
Log.i(TAG, "PARSE FAIL");
|
||||
pageOffset++;
|
||||
} else {
|
||||
mRecordList.add(record);
|
||||
pageOffset+=record.getLength();
|
||||
}
|
||||
}
|
||||
ArrayList<Record> pickyRecords = new ArrayList<>();
|
||||
pickyRecords.addAll(mRecordList);
|
||||
parseByDates(rawPage,model);
|
||||
for (Record r : mRecordList) {
|
||||
for (Record r2 : pickyRecords) {
|
||||
if (r.getFoundAtOffset() == r2.getFoundAtOffset()) {
|
||||
Log.v(TAG,"Found matching record at offset " + r.getFoundAtOffset());
|
||||
}
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean parseByDates(byte[] rawPage, PumpModel model) {
|
||||
mRecordList = new ArrayList<>();
|
||||
if (rawPage.length != 1024) {
|
||||
Log.e(TAG,"Unexpected page size. Expected: 1024 Was: " + rawPage.length);
|
||||
//return false;
|
||||
}
|
||||
this.model = model;
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, "Parsing page");
|
||||
}
|
||||
|
||||
if (rawPage.length < 4) {
|
||||
Log.e(TAG,"Page too short, need at least 4 bytes");
|
||||
return false;
|
||||
}
|
||||
|
||||
this.data = Arrays.copyOfRange(rawPage, 0, rawPage.length-2);
|
||||
this.crc = Arrays.copyOfRange(rawPage, rawPage.length-2, rawPage.length);
|
||||
byte[] expectedCrc = CRC.calculate16CCITT(this.data);
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, String.format("Data length: %d", data.length));
|
||||
}
|
||||
if (!Arrays.equals(crc, expectedCrc)) {
|
||||
Log.w(TAG, String.format("CRC does not match expected value. Expected: %s Was: %s", HexDump.toHexString(expectedCrc), HexDump.toHexString(crc)));
|
||||
} else {
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, "CRC OK");
|
||||
}
|
||||
}
|
||||
|
||||
int pageOffset = 0;
|
||||
PumpTimeStamp lastPumpTimeStamp = new PumpTimeStamp();
|
||||
while (pageOffset < this.data.length - 7) {
|
||||
PumpTimeStamp timestamp = collectTimeStamp(data,pageOffset+2);
|
||||
if (timestamp!=null) {
|
||||
String year = timestamp.toString().substring(0,3);
|
||||
Record record;
|
||||
if ("201".equals(year)) {
|
||||
// maybe found a record.
|
||||
try {
|
||||
record = attemptParseRecord(data, pageOffset);
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
record = null;
|
||||
}
|
||||
if (record != null) {
|
||||
if (timestamp.getLocalDateTime().compareTo(lastPumpTimeStamp.getLocalDateTime()) >= 0) {
|
||||
Log.i(TAG, "Timestamp is increasing");
|
||||
lastPumpTimeStamp = timestamp;
|
||||
mRecordList.add(record);
|
||||
} else {
|
||||
Log.e(TAG, "Timestamp is decreasing");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
pageOffset++;
|
||||
}
|
||||
|
||||
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean parseFrom(byte[] rawPage, PumpModel model) {
|
||||
mRecordList = new ArrayList<>(); // wipe old contents each time when parsing.
|
||||
if (rawPage.length != 1024) {
|
||||
Log.e(TAG,"Unexpected page size. Expected: 1024 Was: " + rawPage.length);
|
||||
//return false;
|
||||
}
|
||||
this.model = model;
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, "Parsing page");
|
||||
}
|
||||
|
||||
if (rawPage.length < 4) {
|
||||
Log.e(TAG,"Page too short, need at least 4 bytes");
|
||||
return false;
|
||||
}
|
||||
|
||||
this.data = Arrays.copyOfRange(rawPage, 0, rawPage.length-2);
|
||||
this.crc = Arrays.copyOfRange(rawPage, rawPage.length-2, rawPage.length);
|
||||
byte[] expectedCrc = CRC.calculate16CCITT(this.data);
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, String.format("Data length: %d", data.length));
|
||||
}
|
||||
if (!Arrays.equals(crc, expectedCrc)) {
|
||||
Log.w(TAG, String.format("CRC does not match expected value. Expected: %s Was: %s", HexDump.toHexString(expectedCrc), HexDump.toHexString(crc)));
|
||||
} else {
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, "CRC OK");
|
||||
}
|
||||
}
|
||||
|
||||
int dataIndex = 0;
|
||||
boolean done = false;
|
||||
while (!done) {
|
||||
Record record = null;
|
||||
if (data[dataIndex] != 0) {
|
||||
// If the data byte is zero, assume that means end of page
|
||||
try {
|
||||
record = attemptParseRecord(data, dataIndex);
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
record = null;
|
||||
}
|
||||
} else {
|
||||
Log.v(TAG,"Zero opcode encountered -- end of page. " + (rawPage.length - dataIndex) + " bytes remaining.");
|
||||
break;
|
||||
}
|
||||
if (record != null) {
|
||||
Log.v(TAG,"parseFrom: found event "+record.getClass().getSimpleName() + " length=" + record.getLength() + " offset=" + record.getFoundAtOffset());
|
||||
mRecordList.add(record);
|
||||
dataIndex += record.getLength();
|
||||
} else {
|
||||
Log.e(TAG,String.format("parseFrom: Failed to parse opcode 0x%02x, offset=%d",data[dataIndex],dataIndex));
|
||||
done = true;
|
||||
}
|
||||
if (dataIndex >= data.length - 2) {
|
||||
done = true;
|
||||
}
|
||||
}
|
||||
if (DEBUG_PAGE) {
|
||||
Log.i(TAG, String.format("Number of records: %d", mRecordList.size()));
|
||||
int index = 1;
|
||||
for (Record r : mRecordList) {
|
||||
Log.v(TAG, String.format("Record #%d: %s", index,r.getShortTypeName()));
|
||||
index += 1;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/* attemptParseRecord will attempt to create a subclass of Record from the given
|
||||
* data and offset. It will return NULL if it fails. If it succeeds, the returned
|
||||
* subclass of Record can be examined for its length, so that the next attempt can be made.
|
||||
*/
|
||||
public static <T extends Record> T attemptParseRecord(byte[] data, int offsetStart) {
|
||||
// no data?
|
||||
if (data == null) {
|
||||
return null;
|
||||
}
|
||||
// invalid offset?
|
||||
if (data.length < offsetStart) {
|
||||
return null;
|
||||
}
|
||||
//Log.d(TAG,String.format("checking for handler for record type 0x%02X at index %d",data[offsetStart],offsetStart));
|
||||
RecordTypeEnum en = RecordTypeEnum.fromByte(data[offsetStart]);
|
||||
T record = en.getRecordClassInstance(PumpModel.MM522);
|
||||
if (record != null) {
|
||||
// have to do this to set the record's opCode
|
||||
byte[] tmpData = new byte[data.length];
|
||||
System.arraycopy(data, offsetStart, tmpData, 0, data.length - offsetStart);
|
||||
boolean didParse = record.parseWithOffset(tmpData, PumpModel.MM522, offsetStart);
|
||||
if (!didParse) {
|
||||
Log.e(TAG,String.format("attemptParseRecord: class %s (opcode 0x%02X) failed to parse at offset %d",record.getShortTypeName(),data[offsetStart],offsetStart));
|
||||
}
|
||||
}
|
||||
return record;
|
||||
}
|
||||
|
||||
public static DateTime parseSimpleDate(byte[] data, int offset) {
|
||||
DateTime timeStamp = null;
|
||||
int seconds = 0;
|
||||
int minutes = 0;
|
||||
int hour = 0;
|
||||
//int high = data[0] >> 4;
|
||||
int low = data[0 + offset] & 0x1F;
|
||||
//int year_high = data[1] >> 4;
|
||||
int mhigh = (data[0 + offset] & 0xE0) >> 4;
|
||||
int mlow = (data[1 + offset] & 0x80) >> 7;
|
||||
int month = mhigh + mlow;
|
||||
int dayOfMonth = low + 1;
|
||||
// python code says year is data[1] & 0x0F, but that will cause problem in 2016.
|
||||
// Hopefully, the remaining bits are part of the year...
|
||||
int year = data[1 + offset] & 0x3F;
|
||||
/*
|
||||
Log.w(TAG, String.format("Attempting to create DateTime from: %04d-%02d-%02d %02d:%02d:%02d",
|
||||
year + 2000, month, dayOfMonth, hour, minutes, seconds));
|
||||
*/
|
||||
try {
|
||||
timeStamp = new DateTime(year + 2000, month, dayOfMonth, hour, minutes, seconds);
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
//Log.e(TAG,"Illegal DateTime field");
|
||||
//e.printStackTrace();
|
||||
return null;
|
||||
}
|
||||
return timeStamp;
|
||||
}
|
||||
|
||||
public static void discoverRecords(byte[] data) {
|
||||
int i = 0;
|
||||
boolean done = false;
|
||||
|
||||
ArrayList<Integer> keyLocations= new ArrayList();
|
||||
while (!done) {
|
||||
RecordTypeEnum en = RecordTypeEnum.fromByte(data[i]);
|
||||
if (en != RecordTypeEnum.RECORD_TYPE_NULL) {
|
||||
keyLocations.add(i);
|
||||
Log.v(TAG, String.format("Possible record of type %s found at index %d", en, i));
|
||||
}
|
||||
/*
|
||||
DateTime ts = parseSimpleDate(data,i);
|
||||
if (ts != null) {
|
||||
if (ts.year().get() == 2015) {
|
||||
Log.w(TAG, String.format("Possible simple date at index %d", i));
|
||||
}
|
||||
}
|
||||
*/
|
||||
i = i + 1;
|
||||
done = (i >= data.length-2);
|
||||
}
|
||||
// for each of the discovered key locations, attempt to parse a sequence of records
|
||||
for(RecordTypeEnum en : RecordTypeEnum.values()) {
|
||||
|
||||
}
|
||||
for (int ix = 0; ix < keyLocations.size(); ix++) {
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
*
|
||||
* For IPC serialization
|
||||
*
|
||||
*/
|
||||
|
||||
/*
|
||||
private byte[] crc;
|
||||
private byte[] data;
|
||||
protected PumpModel model;
|
||||
public List<Record> mRecordList;
|
||||
*/
|
||||
|
||||
public Bundle pack() {
|
||||
Bundle bundle = new Bundle();
|
||||
bundle.putByteArray("crc",crc);
|
||||
bundle.putByteArray("data",data);
|
||||
bundle.putString("model",PumpModel.toString(model));
|
||||
ArrayList<Bundle> records = new ArrayList<>();
|
||||
for (int i=0; i<mRecordList.size(); i++) {
|
||||
try {
|
||||
records.add(mRecordList.get(i).dictionaryRepresentation());
|
||||
} catch (NullPointerException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
bundle.putParcelableArrayList("mRecordList",records);
|
||||
return bundle;
|
||||
}
|
||||
|
||||
public void unpack(Bundle in) {
|
||||
crc = in.getByteArray("crc");
|
||||
data = in.getByteArray("data");
|
||||
model = PumpModel.fromString(in.getString("model"));
|
||||
ArrayList<Bundle> records = in.getParcelableArrayList("mRecordList");
|
||||
mRecordList = new ArrayList<>();
|
||||
if (records != null) {
|
||||
for (int i=0; i<records.size(); i++) {
|
||||
Record r = RecordTypeEnum.getRecordClassInstance(records.get(i),model);
|
||||
r.readFromBundle(records.get(i));
|
||||
mRecordList.add(r);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,51 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import android.content.ContentValues;
|
||||
import android.os.Bundle;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/17/16.
|
||||
*/
|
||||
public class PumpHistoryDatabaseEntry {
|
||||
public static final String key_timestamp = "timestamp";
|
||||
private String timestampString;
|
||||
public static final String key_pageNum = "pagenum";
|
||||
private int pageNum;
|
||||
public static final String key_pageOffset = "offset";
|
||||
private int pageOffset;
|
||||
public static final String key_recordType = "type";
|
||||
private String recordType;
|
||||
public static final String key_length = "length";
|
||||
private int length;
|
||||
public static String getTableInitString() {
|
||||
String rval = "(";
|
||||
rval += "id INTEGER PRIMARY KEY" + ", ";
|
||||
rval += key_timestamp + " TEXT" + ",";
|
||||
rval += key_pageNum + " INTEGER" + ",";
|
||||
rval += key_pageOffset + " INTEGER" + ",";
|
||||
rval += key_recordType + " TEXT" + ",";
|
||||
rval += key_length + " INTEGER" +")";
|
||||
return rval;
|
||||
}
|
||||
public PumpHistoryDatabaseEntry() {
|
||||
}
|
||||
public boolean initFromRecordBundle(int pageNum, Bundle bundle) {
|
||||
timestampString = bundle.getString("timestamp","00-00-00T00:00:00");
|
||||
this.pageNum = pageNum;
|
||||
pageOffset = bundle.getInt("foundAtOffset",-1);
|
||||
recordType = bundle.getString("_type","(unknown)");
|
||||
this.length = bundle.getInt("length",0);
|
||||
return true;
|
||||
}
|
||||
public ContentValues getContentValues() {
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(key_length,length);
|
||||
values.put(key_pageNum,pageNum);
|
||||
values.put(key_pageOffset,pageOffset);
|
||||
values.put(key_recordType,recordType);
|
||||
values.put(key_timestamp,timestampString);
|
||||
return values;
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,57 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.database.sqlite.SQLiteOpenHelper;
|
||||
import android.util.Log;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/17/16.
|
||||
*/
|
||||
public class PumpHistoryDatabaseHandler extends SQLiteOpenHelper {
|
||||
private static final String TAG = "PumpHistoryDatabase";
|
||||
private static final int DATABASE_VERSION = 1;
|
||||
private static final String DATABASE_NAME = "RT2PumpHistory";
|
||||
private static final String DATABASE_TABLE_entries = "entries";
|
||||
|
||||
public PumpHistoryDatabaseHandler(Context context) {
|
||||
super(context,DATABASE_NAME,null,DATABASE_VERSION);
|
||||
}
|
||||
@Override
|
||||
public void onCreate(SQLiteDatabase sqLiteDatabase) {
|
||||
String CREATE_TABLE_entries = "CREATE TABLE " + DATABASE_TABLE_entries + PumpHistoryDatabaseEntry.getTableInitString();
|
||||
sqLiteDatabase.execSQL(CREATE_TABLE_entries);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onUpgrade(SQLiteDatabase sqLiteDatabase, int old_version, int new_version) {
|
||||
sqLiteDatabase.execSQL("DROP TABLE IF EXISTS "+DATABASE_TABLE_entries);
|
||||
onCreate(sqLiteDatabase);
|
||||
}
|
||||
|
||||
public void addEntry(PumpHistoryDatabaseEntry entry) {
|
||||
SQLiteDatabase db = getWritableDatabase();
|
||||
ContentValues values = entry.getContentValues();
|
||||
db.insert(DATABASE_TABLE_entries, null, values);
|
||||
db.close();
|
||||
}
|
||||
|
||||
public void addContentValuesList(List<ContentValues> list) {
|
||||
SQLiteDatabase db = getWritableDatabase();
|
||||
for (ContentValues cvs : list) {
|
||||
db.insert(DATABASE_TABLE_entries, null, cvs);
|
||||
}
|
||||
db.close();
|
||||
Log.d(TAG,"Database "+ DATABASE_NAME + " saved");
|
||||
}
|
||||
|
||||
public void clearPumpHistoryDatabase() {
|
||||
SQLiteDatabase db = getWritableDatabase();
|
||||
db.execSQL("DROP TABLE IF EXISTS "+DATABASE_TABLE_entries);
|
||||
onCreate(db);
|
||||
db.close();
|
||||
}
|
||||
}
|
|
@ -1,341 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.os.Bundle;
|
||||
import android.os.Environment;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RT2Const;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records.Record;
|
||||
import com.gxwtech.roundtrip2.util.StringUtil;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.OutputStream;
|
||||
import java.lang.reflect.Array;
|
||||
import java.util.ArrayList;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/17/16.
|
||||
*/
|
||||
public class PumpHistoryManager {
|
||||
private static final String TAG = "PumpHistoryManager";
|
||||
private Context context;
|
||||
private PumpHistoryDatabaseHandler phdb;
|
||||
ArrayList<ContentValues> dbContentValues = new ArrayList<>();
|
||||
ArrayList<Bundle> packedPages = null;
|
||||
public PumpHistoryManager(Context context) {
|
||||
this.context = context;
|
||||
phdb = new PumpHistoryDatabaseHandler(context);
|
||||
}
|
||||
|
||||
public void initFromPages(Bundle historyBundle) {
|
||||
packedPages = historyBundle.getParcelableArrayList(RT2Const.IPC.MSG_PUMP_history_key);
|
||||
for (int i=0; i<packedPages.size(); i++) {
|
||||
Bundle pageBundle = packedPages.get(i);
|
||||
if (pageBundle != null) {
|
||||
ArrayList<Bundle> recordBundleList = pageBundle.getParcelableArrayList("mRecordList");
|
||||
for (Bundle b : recordBundleList) {
|
||||
try {
|
||||
PumpHistoryDatabaseEntry entry = new PumpHistoryDatabaseEntry();
|
||||
if (entry.initFromRecordBundle(i, b)) {
|
||||
dbContentValues.add(entry.getContentValues());
|
||||
}
|
||||
} catch (java.lang.NullPointerException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
phdb.addContentValuesList(dbContentValues);
|
||||
}
|
||||
|
||||
public void clearDatabase() {
|
||||
phdb.clearPumpHistoryDatabase();
|
||||
}
|
||||
|
||||
/* Checks if external storage is available for read and write */
|
||||
public boolean isExternalStorageWritable() {
|
||||
String state = Environment.getExternalStorageState();
|
||||
if (Environment.MEDIA_MOUNTED.equals(state)) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public boolean timestampOK(String timestamp) {
|
||||
if (timestamp == null) {
|
||||
return false;
|
||||
}
|
||||
if (timestamp.length() < 4) {
|
||||
return false;
|
||||
}
|
||||
if ("2015".equals(timestamp.substring(0,4))) {
|
||||
return true;
|
||||
}
|
||||
if ("2016".equals(timestamp.substring(0,4))) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public HtmlCodeTagStart renderContexts(ArrayList<Bundle> relevantBundles) {
|
||||
// search for all contexts that span this byte
|
||||
// make a start tag that represents them.
|
||||
StringBuilder titleBuilder = new StringBuilder();
|
||||
for (Bundle b : relevantBundles) {
|
||||
String name = b.getString("_type");
|
||||
String timestamp = b.getString("timestamp");
|
||||
int length = b.getInt("length");
|
||||
int offset = b.getInt("foundAtOffset");
|
||||
titleBuilder.append(String.format("[%s%s %s l=%d o=%d]",
|
||||
timestampOK(timestamp) ? "" : "BAD ",
|
||||
name==null?"(null)":name,
|
||||
timestamp == null?"(null)":timestamp,
|
||||
length, offset));
|
||||
}
|
||||
String colorString = null;
|
||||
if (relevantBundles.size() == 1) {
|
||||
if (timestampOK(relevantBundles.get(0).getString("timestamp"))) {
|
||||
colorString = "#99ff99";
|
||||
} else {
|
||||
colorString = "#F0E68C";
|
||||
}
|
||||
} else if (relevantBundles.size() > 1) {
|
||||
boolean allOK = relevantBundlesAllOK(relevantBundles);
|
||||
boolean allBad = relevantBundlesAllBad(relevantBundles);
|
||||
if (allOK) {
|
||||
colorString = "#248f24";
|
||||
} else if (allBad) {
|
||||
colorString = "#cc0000";
|
||||
} else {
|
||||
colorString = "#b3b300";
|
||||
}
|
||||
}
|
||||
return new HtmlCodeTagStart(titleBuilder.toString(),colorString);
|
||||
}
|
||||
|
||||
public ArrayList<Bundle> findRelevantBundles(int pageNum, int pageOffset) {
|
||||
ArrayList<Bundle> relevantBundles = new ArrayList<>();
|
||||
ArrayList<Bundle> recordBundleList = packedPages.get(pageNum).getParcelableArrayList("mRecordList");
|
||||
for (int i=0; i< recordBundleList.size(); i++) {
|
||||
Bundle recordBundle = recordBundleList.get(i);
|
||||
int offset = recordBundle.getInt("foundAtOffset");
|
||||
int length = recordBundle.getInt("length");
|
||||
if ((offset <= pageOffset) && (offset + length > pageOffset)) {
|
||||
relevantBundles.add(recordBundle);
|
||||
}
|
||||
}
|
||||
return relevantBundles;
|
||||
}
|
||||
|
||||
public boolean relevantBundlesAllOK(ArrayList<Bundle> bundles) {
|
||||
for (Bundle b : bundles) {
|
||||
String ts = b.getString("timestamp");
|
||||
if (ts == null) {
|
||||
return false;
|
||||
}
|
||||
if (!timestampOK(ts)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean relevantBundlesAllBad(ArrayList<Bundle> bundles) {
|
||||
for (Bundle b : bundles) {
|
||||
String ts = b.getString("timestamp");
|
||||
if (ts != null) {
|
||||
if (timestampOK(ts)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean relevantBundlesSame(ArrayList<Bundle> bundles1, ArrayList<Bundle> bundles2) {
|
||||
if (bundles1.size() != bundles2.size()) {
|
||||
return false;
|
||||
}
|
||||
for (Bundle b : bundles1) {
|
||||
if (!bundles2.contains(b)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
for (Bundle b : bundles2) {
|
||||
if (!bundles1.contains(b)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean relevantBundlesAllDifferent(ArrayList<Bundle> bundles1, ArrayList<Bundle> bundles2) {
|
||||
for (Bundle b : bundles1) {
|
||||
if (bundles2.contains(b)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
for (Bundle b : bundles2) {
|
||||
if (bundles1.contains(b)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public ArrayList<HtmlElement> makeDom() {
|
||||
ArrayList<HtmlElement> rval = new ArrayList<>();
|
||||
for (int pageNum = 0; pageNum < packedPages.size(); pageNum++) {
|
||||
Log.i(TAG,"Rendering page " + pageNum);
|
||||
rval.add(new HtmlHistoryPageStart(pageNum));
|
||||
byte[] pageData = packedPages.get(pageNum).getByteArray("data");
|
||||
if (pageData == null) {
|
||||
return rval;
|
||||
}
|
||||
int pageSize = pageData.length;
|
||||
byte[] crc = packedPages.get(pageNum).getByteArray("crc");
|
||||
if (pageSize != 1022) {
|
||||
Log.e(TAG, "Page size is not 1022, it is " + pageSize);
|
||||
}
|
||||
int pageOffset = 0;
|
||||
boolean done = false;
|
||||
ArrayList<Bundle> currentBundles = findRelevantBundles(pageNum, pageOffset);
|
||||
ArrayList<Bundle> nextBundles = new ArrayList<>();
|
||||
rval.add(renderContexts(currentBundles));
|
||||
while (!done) {
|
||||
rval.add(new HtmlByte(pageData[pageOffset]));
|
||||
if (pageOffset == pageSize - 1) {
|
||||
done = true;
|
||||
} else {
|
||||
nextBundles = findRelevantBundles(pageNum, pageOffset + 1);
|
||||
if (relevantBundlesSame(currentBundles, nextBundles)) {
|
||||
// Not changing context
|
||||
if (pageOffset % 32 == 31) {
|
||||
rval.add(new HtmlElementGeneric("<br>\n"));
|
||||
} else {
|
||||
rval.add(new HtmlElementGeneric(" "));
|
||||
}
|
||||
} else {
|
||||
// Changing context
|
||||
rval.add(new HtmlCodeTagEnd());
|
||||
if (relevantBundlesAllDifferent(currentBundles, nextBundles)) {
|
||||
rval.add(new HtmlCodeTagStart("", "")); // <code>
|
||||
if (pageOffset % 32 == 31) {
|
||||
rval.add(new HtmlElementGeneric("<br>\n"));
|
||||
} else {
|
||||
rval.add(new HtmlElementGeneric(" "));
|
||||
}
|
||||
if (nextBundles.size() != 0) {
|
||||
rval.add(new HtmlCodeTagEnd());
|
||||
rval.add(renderContexts(nextBundles));
|
||||
}
|
||||
} else {
|
||||
if (pageOffset % 32 == 31) {
|
||||
rval.add(new HtmlElementGeneric("<br>\n"));
|
||||
} else {
|
||||
rval.add(new HtmlElementGeneric(" "));
|
||||
}
|
||||
rval.add(new HtmlCodeTagEnd());
|
||||
rval.add(renderContexts(nextBundles));
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!done) {
|
||||
pageOffset++;
|
||||
currentBundles = nextBundles;
|
||||
}
|
||||
}
|
||||
}
|
||||
return rval;
|
||||
}
|
||||
|
||||
public void writeHtmlPage() {
|
||||
|
||||
/*
|
||||
final String key_timestamp = "timestamp";
|
||||
String timestampString;
|
||||
final String key_pageNum = "pagenum";
|
||||
int pageNum;
|
||||
final String key_pageOffset = "offset";
|
||||
int foundAtOffset;
|
||||
final String key_recordType = "type";
|
||||
String recordType;
|
||||
final String key_length = "length";
|
||||
int length;
|
||||
*/
|
||||
if (!isExternalStorageWritable()) {
|
||||
Log.e(TAG,"External storage not writable.");
|
||||
return;
|
||||
}
|
||||
|
||||
String filename = "PumpHistoryBytes.html";
|
||||
|
||||
File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_MUSIC);
|
||||
File pageFile = new File(path,filename);
|
||||
|
||||
OutputStream os = null;
|
||||
try {
|
||||
os = new FileOutputStream(pageFile);
|
||||
} catch (FileNotFoundException fnf) {
|
||||
Log.e(TAG,"Failed to open " + filename + " for writing");
|
||||
}
|
||||
if (os == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
// write header
|
||||
try {
|
||||
os.write("<!DOCTYPE html>".getBytes());
|
||||
os.write("<html>".getBytes());
|
||||
os.write("<head>".getBytes());
|
||||
os.write("<title>Page Title</title>".getBytes());
|
||||
os.write("</head>".getBytes());
|
||||
os.write("<body>".getBytes());
|
||||
|
||||
byte[] pageData = packedPages.get(0).getByteArray("data");
|
||||
int pageSize = pageData.length;
|
||||
//byte[] crc = packedPages.get(0).getByteArray("crc");
|
||||
if (pageSize != 1022) {
|
||||
Log.e(TAG,"Page size is not 1022, it is " + pageSize);
|
||||
}
|
||||
|
||||
ArrayList<HtmlElement> dom = makeDom();
|
||||
Log.i(TAG,"There are " + dom.size() + " elements to render.");
|
||||
for (HtmlElement e : dom) {
|
||||
if (e != null) {
|
||||
String elementString = e.toString();
|
||||
if (elementString != null) {
|
||||
byte[] bytes = elementString.getBytes();
|
||||
if (bytes != null) {
|
||||
os.write(bytes);
|
||||
} else {
|
||||
Log.e(TAG,"WriteHtmlPage: bytes is null");
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG,"WriteHtmlPage: elementString is null");
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG,"WriteHtmlPage: element is null");
|
||||
}
|
||||
}
|
||||
|
||||
os.write("</body></html>".getBytes());
|
||||
os.close();
|
||||
} catch (FileNotFoundException fnf) {
|
||||
fnf.printStackTrace();
|
||||
} catch (IOException ioe) {
|
||||
ioe.printStackTrace();
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,43 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
import com.gxwtech.roundtrip2.util.CRC;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/18/16.
|
||||
*/
|
||||
public class PumpHistoryPage {
|
||||
public byte[] data = new byte[0];
|
||||
private PumpModel model = PumpModel.MM522;
|
||||
private int pageNumber;
|
||||
public PumpHistoryPage() {}
|
||||
public PumpHistoryPage(byte[] data, PumpModel model, int pageNumber) {
|
||||
init(data,model,pageNumber);
|
||||
}
|
||||
public void init(byte[] data, PumpModel model, int pageNumber) {
|
||||
this.data = data;
|
||||
this.model = model;
|
||||
this.pageNumber = pageNumber;
|
||||
}
|
||||
public int getPageNumber() { return pageNumber; }
|
||||
public boolean isValid() {
|
||||
return isCRCValid();
|
||||
}
|
||||
public boolean isCRCValid() {
|
||||
if (data == null) {
|
||||
return false;
|
||||
}
|
||||
if (data.length < 3) {
|
||||
return false;
|
||||
}
|
||||
byte[] crc16 = CRC.calculate16CCITT(ByteUtil.substring(data,0,data.length-2));
|
||||
if (crc16 == null) {
|
||||
return false;
|
||||
}
|
||||
if ((crc16[0] == data[data.length-2]) && (crc16[1]==data[data.length-1])) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
|
@ -1,263 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
import android.os.Bundle;
|
||||
import android.util.Log;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpTimeStamp;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.TimeFormat;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/18/16.
|
||||
*
|
||||
*
|
||||
*
|
||||
* This class is not in use -- it may become a replacement for the history object heirarchy parser
|
||||
*
|
||||
*
|
||||
*
|
||||
*
|
||||
*
|
||||
*
|
||||
*
|
||||
*/
|
||||
@Deprecated
|
||||
public class PumpHistoryParser {
|
||||
private static final String TAG = "PumpHistoryParser";
|
||||
public PumpModel model;
|
||||
public ArrayList<PumpHistoryPage> pages = new ArrayList<>();
|
||||
public ArrayList<Bundle> entries = new ArrayList<>();
|
||||
public PumpHistoryParser(PumpModel model) {
|
||||
initOpcodes();
|
||||
}
|
||||
public void parsePage(byte[] rawData) {
|
||||
PumpHistoryPage page = new PumpHistoryPage(rawData,model,pages.size()+1);
|
||||
pages.add(page);
|
||||
if (page.isCRCValid() == false) {
|
||||
Log.e(TAG,"CRC16 for page " + page.getPageNumber() + " is invalid.");
|
||||
}
|
||||
if (page.data.length != 1024) {
|
||||
Log.w(TAG,String.format("Page %d has length %d, expected 1024",page.getPageNumber(),page.data.length));
|
||||
}
|
||||
int currentOffset = 0;
|
||||
boolean done = false;
|
||||
while (!done) {
|
||||
Bundle b = attemptParseRecord(page.getPageNumber(),currentOffset);
|
||||
if (b == null) {
|
||||
// then the parse failed.
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int asUINT8(byte b) {
|
||||
return (b < 0) ? b + 256 : b;
|
||||
}
|
||||
|
||||
static double insulinDecode(int a, int b) {
|
||||
return ((a << 8) + b) / 40.0;
|
||||
}
|
||||
|
||||
public Bundle attemptParseRecord(int pageNum, int offset) {
|
||||
byte[] dataPage = pages.get(pageNum).data;
|
||||
int opcode = pages.get(pageNum).data[offset];
|
||||
Bundle record = new Bundle();
|
||||
String typename = opCodeNames.get(opcode);
|
||||
if (typename != null) {
|
||||
record.putInt("opcode",opcode);
|
||||
record.putString("type",typename);
|
||||
record.putInt("page",pageNum);
|
||||
record.putInt("offset",offset);
|
||||
record.putString("indexer",String.format("%02d%04d",pageNum,offset));
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
int length = 0;
|
||||
int dateOffset = 0;
|
||||
PumpTimeStamp timestamp;
|
||||
byte[] data = ByteUtil.substring(dataPage,offset,dataPage.length - 2 - offset);
|
||||
switch (opcode) {
|
||||
case RECORD_TYPE_BolusNormal: {
|
||||
length = PumpModel.isLargerFormat(model) ? 13 : 9;
|
||||
if (length + offset > data.length) return null;
|
||||
record.putInt("length",length);
|
||||
double programmedAmount;
|
||||
double deliveredAmount;
|
||||
double unabsorbedInsulinTotal;
|
||||
int duration;
|
||||
String bolusType;
|
||||
if (PumpModel.isLargerFormat(model)) {
|
||||
programmedAmount = insulinDecode(asUINT8(data[1]),asUINT8(data[2]));
|
||||
deliveredAmount = insulinDecode(asUINT8(data[3]),asUINT8(data[4]));
|
||||
unabsorbedInsulinTotal = insulinDecode(asUINT8(data[5]),asUINT8(data[6]));
|
||||
duration = asUINT8(data[7]) * 30;
|
||||
try {
|
||||
timestamp = new PumpTimeStamp(TimeFormat.parse5ByteDate(data, 8));
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
timestamp = new PumpTimeStamp();
|
||||
}
|
||||
} else {
|
||||
programmedAmount = asUINT8(data[1]) / 10.0f;
|
||||
deliveredAmount = asUINT8(data[2]) / 10.0f;
|
||||
duration = asUINT8(data[3]) * 30;
|
||||
unabsorbedInsulinTotal = 0;
|
||||
try {
|
||||
timestamp = new PumpTimeStamp(TimeFormat.parse5ByteDate(data, 4));
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
timestamp = new PumpTimeStamp();
|
||||
}
|
||||
}
|
||||
bolusType = (duration > 0) ? "square" : "normal";
|
||||
record.putDouble("programmedAmount",programmedAmount);
|
||||
record.putDouble("deliveredAmount",deliveredAmount);
|
||||
record.putInt("duration",duration);
|
||||
record.putDouble("unabsorbedInsulinTotal",unabsorbedInsulinTotal);
|
||||
record.putString("bolusType",bolusType);
|
||||
record.putString("timestamp",timestamp.getLocalDateTime().toString());
|
||||
|
||||
} break;
|
||||
case RECORD_TYPE_AlarmSensor:
|
||||
length = 8;
|
||||
if (length + offset > data.length) return null;
|
||||
record.putInt("length",length);
|
||||
case RECORD_TYPE_Resume:
|
||||
case RECORD_TYPE_AlarmClockReminder:
|
||||
length = 7;
|
||||
if (length + offset > data.length) return null;
|
||||
record.putInt("length",length);
|
||||
try {
|
||||
timestamp = new PumpTimeStamp(TimeFormat.parse5ByteDate(data, 2));
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
timestamp = new PumpTimeStamp();
|
||||
}
|
||||
record.putString("timestamp",timestamp.getLocalDateTime().toString());
|
||||
default:
|
||||
record.putInt("length",0);
|
||||
record.putString("comment","unparsed");
|
||||
}
|
||||
return record;
|
||||
}
|
||||
|
||||
HashMap<Integer,String> opCodeNames = new HashMap<>();
|
||||
HashMap<String,Integer> opCodeNumbers = new HashMap<>();
|
||||
|
||||
private void initOpcode(int number, String name) {
|
||||
opCodeNames.put(number,name);
|
||||
opCodeNumbers.put(name,number);
|
||||
}
|
||||
|
||||
public static final int RECORD_TYPE_BolusNormal = 0x01;
|
||||
public static final int RECORD_TYPE_Prime = 0x03;
|
||||
public static final int RECORD_TYPE_PumpAlarm = 0x06;
|
||||
public static final int RECORD_TYPE_ResultDailyTotal = 0x07;
|
||||
public static final int RECORD_TYPE_ChangeBasalProfilePattern = 0x08;
|
||||
public static final int RECORD_TYPE_ChangeBasalProfile = 0x09;
|
||||
public static final int RECORD_TYPE_CalBgForPh = 0x0A;
|
||||
public static final int RECORD_TYPE_AlarmSensor = 0x0B;
|
||||
public static final int RECORD_TYPE_ClearAlarm = 0x0C;
|
||||
public static final int RECORD_TYPE_SelectBasalProfile = 0x14;
|
||||
public static final int RECORD_TYPE_TempBasalDuration = 0x16;
|
||||
public static final int RECORD_TYPE_ChangeTime = 0x17;
|
||||
public static final int RECORD_TYPE_NewTimeSet = 0x18;
|
||||
public static final int RECORD_TYPE_JournalEntryPumpLowBattery = 0x19;
|
||||
public static final int RECORD_TYPE_Battery = 0x1A;
|
||||
public static final int RECORD_TYPE_Suspend = 0x1E;
|
||||
public static final int RECORD_TYPE_Resume = 0x1F;
|
||||
public static final int RECORD_TYPE_Rewind = 0x21;
|
||||
public static final int RECORD_TYPE_ChangeChildBlockEnable = 0x23;
|
||||
public static final int RECORD_TYPE_ChangeMaxBolus = 0x24;
|
||||
public static final int RECORD_TYPE_EnableDisableRemote = 0x26;
|
||||
public static final int RECORD_TYPE_TempBasalRate = 0x33;
|
||||
public static final int RECORD_TYPE_JournalEntryPumpLowReservoir = 0x34;
|
||||
public static final int RECORD_TYPE_AlarmClockReminder = 0x35;
|
||||
public static final int RECORD_TYPE_BGReceived = 0x3F;
|
||||
public static final int RECORD_TYPE_JournalEntryExerciseMarker = 0x41;
|
||||
public static final int RECORD_TYPE_ChangeSensorSetup2 = 0x50;
|
||||
public static final int RECORD_TYPE_ChangeSensorRateOfChangeAlertSetup = 0x56;
|
||||
public static final int RECORD_TYPE_ChangeBolusScrollStepSize = 0x57;
|
||||
public static final int RECORD_TYPE_ChangeBolusWizardSetup = 0x5A;
|
||||
public static final int RECORD_TYPE_BolusWizardBolusEstimate = 0x5B;
|
||||
public static final int RECORD_TYPE_UnabsorbedInsulin = 0x5C;
|
||||
public static final int RECORD_TYPE_ChangeVariableBolus = 0x5e;
|
||||
public static final int RECORD_TYPE_ChangeAudioBolus = 0x5f;
|
||||
public static final int RECORD_TYPE_ChangeBGReminderEnable = 0x60;
|
||||
public static final int RECORD_TYPE_ChangeAlarmClockEnable = 0x61;
|
||||
public static final int RECORD_TYPE_ChangeTempBasalType = 0x62;
|
||||
public static final int RECORD_TYPE_ChangeAlarmNotifyMode = 0x63;
|
||||
public static final int RECORD_TYPE_ChangeTimeFormat = 0x64;
|
||||
public static final int RECORD_TYPE_ChangeReservoirWarningTime = 0x65;
|
||||
public static final int RECORD_TYPE_ChangeBolusReminderEnable = 0x66;
|
||||
public static final int RECORD_TYPE_ChangeBolusReminderTime = 0x67;
|
||||
public static final int RECORD_TYPE_DeleteBolusReminderTime = 0x68;
|
||||
public static final int RECORD_TYPE_DeleteAlarmClockTime = 0x6a;
|
||||
public static final int RECORD_TYPE_Model522ResultTotals = 0x6D;
|
||||
public static final int RECORD_TYPE_Sara6E = 0x6E;
|
||||
public static final int RECORD_TYPE_ChangeCarbUnits = 0x6f;
|
||||
public static final int RECORD_TYPE_BasalProfileStart = 0x7B;
|
||||
public static final int RECORD_TYPE_ChangeWatchdogEnable = 0x7c;
|
||||
public static final int RECORD_TYPE_ChangeOtherDeviceID = 0x7d;
|
||||
public static final int RECORD_TYPE_ChangeWatchdogMarriageProfile = 0x81;
|
||||
public static final int RECORD_TYPE_DeleteOtherDeviceID = 0x82;
|
||||
public static final int RECORD_TYPE_ChangeCaptureEventEnable = 0x83;
|
||||
|
||||
private void initOpcodes() {
|
||||
initOpcode(0x01,"BolusNormal");
|
||||
initOpcode(0x03,"Prime");
|
||||
initOpcode(0x06,"PumpAlarm");
|
||||
initOpcode(0x07,"ResultDailyTotal");
|
||||
initOpcode(0x08,"ChangeBasalProfilePattern");
|
||||
initOpcode(0x09,"ChangeBasalProfile");
|
||||
initOpcode(0x0A,"CalBgForPh");
|
||||
initOpcode(0x0B,"AlarmSensor");
|
||||
initOpcode(0x0C,"ClearAlarm");
|
||||
//initOpcode(0x14,"SelectBasalProfile");
|
||||
initOpcode(0x16,"TempBasalDuration");
|
||||
initOpcode(0x17,"ChangeTime");
|
||||
//initOpcode(0x18,"NewTimeSet");
|
||||
initOpcode(0x19,"JournalEntryPumpLowBattery");
|
||||
initOpcode(0x1A,"Battery");
|
||||
initOpcode(0x1E,"Suspend");
|
||||
initOpcode(0x1F,"Resume");
|
||||
initOpcode(0x21,"Rewind");
|
||||
initOpcode(0x23,"ChangeChildBlockEnable");
|
||||
initOpcode(0x24,"ChangeMaxBolus");
|
||||
initOpcode(0x26,"EnableDisableRemote");
|
||||
initOpcode(0x33,"TempBasalRate");
|
||||
initOpcode(0x34,"JournalEntryPumpLowReservoir");
|
||||
initOpcode(0x35,"AlarmClockReminder");
|
||||
initOpcode(0x3F,"BGReceived");
|
||||
initOpcode(0x41,"JournalEntryExerciseMarker");
|
||||
initOpcode(0x50,"ChangeSensorSetup2");
|
||||
initOpcode(0x56,"ChangeSensorRateOfChangeAlertSetup");
|
||||
initOpcode(0x57,"ChangeBolusScrollStepSize");
|
||||
initOpcode(0x5A,"ChangeBolusWizardSetup");
|
||||
initOpcode(0x5B,"BolusWizardBolusEstimate");
|
||||
initOpcode(0x5C,"UnabsorbedInsulin");
|
||||
initOpcode(0x5e,"ChangeVariableBolus");
|
||||
initOpcode(0x5f,"ChangeAudioBolus");
|
||||
initOpcode(0x60,"ChangeBGReminderEnable");
|
||||
initOpcode(0x61,"ChangeAlarmClockEnable");
|
||||
initOpcode(0x62,"ChangeTempBasalType");
|
||||
initOpcode(0x63,"ChangeAlarmNotifyMode");
|
||||
initOpcode(0x64,"ChangeTimeFormat");
|
||||
initOpcode(0x65,"ChangeReservoirWarningTime");
|
||||
initOpcode(0x66,"ChangeBolusReminderEnable");
|
||||
initOpcode(0x67,"ChangeBolusReminderTime");
|
||||
initOpcode(0x68,"DeleteBolusReminderTime");
|
||||
initOpcode(0x6a,"DeleteAlarmClockTime");
|
||||
initOpcode(0x6D,"Model522ResultTotals");
|
||||
initOpcode(0x6E,"Sara6E");
|
||||
initOpcode(0x6f,"ChangeCarbUnits");
|
||||
initOpcode(0x7B,"BasalProfileStart");
|
||||
initOpcode(0x7c,"ChangeWatchdogEnable");
|
||||
initOpcode(0x7d,"ChangeOtherDeviceID");
|
||||
initOpcode(0x81,"ChangeWatchdogMarriageProfile");
|
||||
initOpcode(0x82,"DeleteOtherDeviceID");
|
||||
initOpcode(0x83,"ChangeCaptureEventEnable");
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,168 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/7/15.
|
||||
*
|
||||
* This class is intended to be very basic, and only hold the simple data.
|
||||
* That way it will be portable, reusable.
|
||||
*
|
||||
* For instance, for the android UI, there is a parcelable version of this class
|
||||
* (PumpSettingsParcel) but I don't want the pump implementation to be dependent
|
||||
* on Android.Parcelable.
|
||||
*
|
||||
* For marshalling the data into other forms, I'm keeping the original
|
||||
* byte array (mRawData) around. So if we have to save/reload this data object
|
||||
* we can just write the raw data array, and reload from it. We already have to
|
||||
* parse our values from it, so just keep it handy. Makes the Parcelable
|
||||
* implementation very simple.
|
||||
*
|
||||
*/
|
||||
@Deprecated
|
||||
public class PumpSettings {
|
||||
private static final String TAG = "PumpSettings";
|
||||
|
||||
// these values help with the parcel extension
|
||||
protected static final int MINIMUM_DATA_LENGTH = 20; // bytes
|
||||
protected static final int MAXIMUM_DATA_LENGTH = 64; // bytes
|
||||
protected byte[] mRawData;
|
||||
|
||||
public byte mAutoOffDuration_hours = 0;
|
||||
public byte mAlarmMode = 0;
|
||||
public byte mAlarmVolume = 0;
|
||||
public boolean mAudioBolusEnable = false;
|
||||
public double mAudioBolusSize = 0.0;
|
||||
public boolean mVariableBolusEnable = false;
|
||||
public double mMaxBolus = 0.0; // MM23 is different (?)
|
||||
// MM512 and up (?)
|
||||
public double mMaxBasal = 0.0;
|
||||
public byte mTimeFormat = 0;
|
||||
public int mInsulinConcentration = 100; // 100 or 50
|
||||
public boolean mPatternsEnabled = false;
|
||||
public BasalProfileTypeEnum mSelectedPattern = BasalProfileTypeEnum.STD;
|
||||
public boolean mRFEnable = false;
|
||||
public boolean mBlockEnable = false;
|
||||
public byte mTempBasalType = 0; // 1 means Percent, 0 means UnitsPerHour
|
||||
public byte mTempBasalRate = 0; // TODO: scaling?
|
||||
public byte mParadigmEnable = 0;
|
||||
// MM12
|
||||
// boolean mInsulinActionType (0='Fast' or 1='Regular')
|
||||
// MM15
|
||||
public byte mInsulinActionType = 0;
|
||||
public byte mLowReservoirWarnType = 0;
|
||||
public byte mLowReservoirWarnPoint = 0;
|
||||
public byte mKeypadLockStatus = 0;
|
||||
|
||||
public PumpSettings() {
|
||||
init();
|
||||
}
|
||||
|
||||
public void init() {
|
||||
// this data buffer has to be sized to maximum even when empty
|
||||
// as we may be reading from a parcel. See PumpSettingsParcel.
|
||||
mRawData = new byte[MAXIMUM_DATA_LENGTH];
|
||||
/* fill in default values */
|
||||
}
|
||||
|
||||
public byte[] getRawData() { return mRawData; }
|
||||
|
||||
public boolean parseFrom(byte[] data) {
|
||||
if (!setRawData(data)) { return false; }
|
||||
return parseFromRaw();
|
||||
}
|
||||
|
||||
// copy as much as we can from a byte array, but don't parse it.
|
||||
public boolean setRawData(byte[] data) {
|
||||
if (data == null) {
|
||||
return false;
|
||||
}
|
||||
int len = Math.min(MAXIMUM_DATA_LENGTH,data.length);
|
||||
System.arraycopy(data,0,mRawData,0,len);
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean parseFromRaw() {
|
||||
mAutoOffDuration_hours = mRawData[0];
|
||||
mAlarmVolume = mRawData[1];
|
||||
mAlarmMode = 2;
|
||||
if (mRawData[1] == 4) {
|
||||
mAlarmVolume = -1;
|
||||
mAlarmMode = 1;
|
||||
}
|
||||
mAudioBolusEnable = (mRawData[2] == 1);
|
||||
mAudioBolusSize = 0;
|
||||
if (mAudioBolusEnable) {
|
||||
mAudioBolusSize = mRawData[3] / 10.0;
|
||||
}
|
||||
mVariableBolusEnable = (mRawData[4] == 1);
|
||||
// MM23 is different
|
||||
int maxBolusByte = mRawData[5];
|
||||
if (maxBolusByte < 0) { maxBolusByte += 256; }
|
||||
mMaxBolus = (maxBolusByte) / 10.0;
|
||||
// MM512 and up
|
||||
// TODO: check mMaxBasal calculation
|
||||
// did I get the bytes in the right order?
|
||||
int maxBasalHighByte = mRawData[6];
|
||||
if (maxBasalHighByte < 0) {
|
||||
maxBasalHighByte += 256;
|
||||
}
|
||||
int maxBasalLowByte = mRawData[7];
|
||||
if (maxBasalLowByte < 0) {
|
||||
maxBasalLowByte += 256;
|
||||
}
|
||||
mMaxBasal = ((maxBasalHighByte * 256) + maxBasalLowByte)/40.0;
|
||||
mTimeFormat = mRawData[8];
|
||||
// mInsulinConcentration: 0 is 100%, 1 is 50%
|
||||
mInsulinConcentration = 100;
|
||||
if (mRawData[9] == 1) {
|
||||
mInsulinConcentration = 50;
|
||||
}
|
||||
mPatternsEnabled = (mRawData[10] == 1);
|
||||
mSelectedPattern = BasalProfileTypeEnum.STD;
|
||||
if (mRawData[11] == 0x01) {
|
||||
mSelectedPattern = BasalProfileTypeEnum.A;
|
||||
} else if (mRawData[11] == 0x02) {
|
||||
mSelectedPattern = BasalProfileTypeEnum.B;
|
||||
}
|
||||
mRFEnable = (mRawData[12] == 1);
|
||||
mBlockEnable = (mRawData[13] == 1);
|
||||
mTempBasalType = mRawData[14]; // todo: put into proper class of its own
|
||||
mTempBasalRate = mRawData[15];
|
||||
|
||||
mParadigmEnable = mRawData[16];
|
||||
mInsulinActionType = mRawData[17];
|
||||
mLowReservoirWarnType = mRawData[18];
|
||||
mLowReservoirWarnPoint = mRawData[19];
|
||||
mKeypadLockStatus = mRawData[20];
|
||||
return true;
|
||||
}
|
||||
public String explain() {
|
||||
String rval = "";
|
||||
// write out a string describing the current contents
|
||||
rval += String.format("Auto-Off Duration (hours): %d\n",mAutoOffDuration_hours);
|
||||
rval += String.format("Alarm (Volume: %d, Mode: %d)\n",mAlarmVolume,mAlarmMode);
|
||||
rval += String.format("Audio BolusNormalPumpEvent (enabled=%s, size=%g)\n",mAudioBolusEnable,mAudioBolusSize);
|
||||
rval += String.format("Variable BolusNormalPumpEvent Enable: %s\n", mVariableBolusEnable);
|
||||
rval += String.format("Max BolusNormalPumpEvent: %g\n",mMaxBolus);
|
||||
rval += String.format("Max Basal: %g\n",mMaxBasal);
|
||||
rval += String.format("Time Format 0x%02X\n", mTimeFormat);
|
||||
rval += String.format("Insulin Concentration %d%%\n",mInsulinConcentration);
|
||||
rval += String.format("Patterns Enabled: %s\n",mPatternsEnabled);
|
||||
rval += String.format("Selected Pattern: %d\n",mSelectedPattern);
|
||||
rval += String.format("RF Enabled: %s\n",mRFEnable);
|
||||
rval += String.format("Block Enabled: %s\n",mBlockEnable);
|
||||
rval += String.format("Temp Basal Type: %s\n",(mTempBasalType == 1)?"Percent":"Units/Hr");
|
||||
if (mTempBasalType == 1) {
|
||||
rval += String.format("Temp Basal Rate: %d%%\n",mTempBasalRate);
|
||||
} else {
|
||||
rval += String.format("Temp Basal Rate: %d U/h\n",mTempBasalRate);
|
||||
}
|
||||
rval += String.format("Paradigm Enabled: %s\n", mParadigmEnable);
|
||||
rval += String.format("Insulin Action Type: 0x%02X\n",mInsulinActionType);
|
||||
rval += String.format("Low Reservoir Warn Type: 0x%02X\n",mLowReservoirWarnType);
|
||||
rval += String.format("Low Reservoir Warn Point: 0x%02X\n",mLowReservoirWarnPoint);
|
||||
rval += String.format("Keypad Lock Status: 0x%02X\n",mKeypadLockStatus);
|
||||
return rval;
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -1,45 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData;
|
||||
|
||||
/**
|
||||
* Created by geoff on 5/29/15.
|
||||
*
|
||||
* Just need a class to keep the pair together, for parcel transport.
|
||||
*/
|
||||
@Deprecated
|
||||
public class TempBasalPair {
|
||||
private double mInsulinRate = 0.0;
|
||||
private int mDurationMinutes = 0;
|
||||
private boolean mIsPercent = false;
|
||||
|
||||
public double getInsulinRate() {
|
||||
return mInsulinRate;
|
||||
}
|
||||
|
||||
public void setInsulinRate(double insulinRate) {
|
||||
this.mInsulinRate = insulinRate;
|
||||
}
|
||||
|
||||
public int getDurationMinutes() {
|
||||
return mDurationMinutes;
|
||||
}
|
||||
|
||||
public void setDurationMinutes(int durationMinutes) {
|
||||
this.mDurationMinutes = durationMinutes;
|
||||
}
|
||||
|
||||
public boolean isPercent() {
|
||||
return mIsPercent;
|
||||
}
|
||||
|
||||
public void setIsPercent(boolean yesIsPercent) {
|
||||
this.mIsPercent = yesIsPercent;
|
||||
}
|
||||
|
||||
public TempBasalPair() { }
|
||||
|
||||
public TempBasalPair(double insulinRate, boolean isPercent, int durationMinutes) {
|
||||
mInsulinRate = insulinRate;
|
||||
mIsPercent = isPercent;
|
||||
mDurationMinutes = durationMinutes;
|
||||
}
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/11/16.
|
||||
*/
|
||||
public class AlarmClockReminderPumpEvent extends TimeStampedRecord {
|
||||
public AlarmClockReminderPumpEvent() {};
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Alarm Reminder";
|
||||
}
|
||||
}
|
|
@ -1,19 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class AlarmSensorPumpEvent extends TimeStampedRecord {
|
||||
public AlarmSensorPumpEvent() {}
|
||||
|
||||
@Override
|
||||
public int getLength() { return 8; }
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Alarm Sensor";
|
||||
}
|
||||
}
|
|
@ -1,47 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
|
||||
import android.os.Bundle;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.util.ByteUtil;
|
||||
|
||||
public class BGReceivedPumpEvent extends TimeStampedRecord {
|
||||
private int amount = 0;
|
||||
private byte[] meter = new byte[3];
|
||||
|
||||
public BGReceivedPumpEvent() {
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getLength() { return 10; }
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "BG Received";
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean parseFrom(byte[] data, PumpModel model) {
|
||||
if (!super.simpleParse(data,2)) {
|
||||
return false;
|
||||
}
|
||||
amount = (asUINT8(data[1]) << 3) + (asUINT8(data[4])>>5);
|
||||
meter = ByteUtil.substring(data,7,3);
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean readFromBundle(Bundle in) {
|
||||
amount = in.getInt("amount");
|
||||
meter = in.getByteArray("meter");
|
||||
return super.readFromBundle(in);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToBundle(Bundle in) {
|
||||
super.writeToBundle(in);
|
||||
in.putInt("amount",amount);
|
||||
in.putByteArray("meter",meter);
|
||||
}
|
||||
}
|
|
@ -1,51 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
import android.os.Bundle;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
public class BasalProfileStart extends TimeStampedRecord {
|
||||
private static final String TAG = "BasalProfileStart";
|
||||
private int offset = 0;
|
||||
private double rate = 0.0;
|
||||
private int profileIndex = 0;
|
||||
|
||||
public BasalProfileStart() { }
|
||||
|
||||
@Override
|
||||
public int getLength() { return 10; }
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Basal Profile Start";
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean parseFrom(byte[] data, PumpModel model) {
|
||||
if (!simpleParse(data,2)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
profileIndex = asUINT8(data[1]);
|
||||
offset = asUINT8(data[7]) * 30 * 1000 * 60;
|
||||
rate = (double)(asUINT8(data[8])) / 40.0;
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean readFromBundle(Bundle in) {
|
||||
offset = in.getInt("offset");
|
||||
rate = in.getDouble("rate");
|
||||
profileIndex = in.getInt("profileIndex");
|
||||
return super.readFromBundle(in);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToBundle(Bundle in) {
|
||||
super.writeToBundle(in);
|
||||
in.putInt("offset",offset);
|
||||
in.putDouble("rate",rate);
|
||||
in.putInt("profileIndex",profileIndex);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,10 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
public class BatteryPumpEvent extends TimeStampedRecord {
|
||||
public BatteryPumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Battery";
|
||||
}
|
||||
}
|
|
@ -1,86 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
|
||||
import android.os.Bundle;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpTimeStamp;
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.TimeFormat;
|
||||
|
||||
public class BolusNormalPumpEvent extends TimeStampedRecord {
|
||||
private final static String TAG = "BolusNormalPumpEvent";
|
||||
|
||||
private double programmedAmount = 0.0;
|
||||
private double deliveredAmount = 0.0;
|
||||
private int duration = 0;
|
||||
private double unabsorbedInsulinTotal = 0.0;
|
||||
private String bolusType = "Unset";
|
||||
|
||||
public BolusNormalPumpEvent() {
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getLength() { return PumpModel.isLargerFormat(model) ? 13 : 9; }
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Normal Bolus";
|
||||
}
|
||||
|
||||
private double insulinDecode(int a, int b) {
|
||||
return ((a << 8) + b) / 40.0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean parseFrom(byte[] data, PumpModel model) {
|
||||
if (getLength() > data.length) {
|
||||
return false;
|
||||
}
|
||||
if (PumpModel.isLargerFormat(model)) {
|
||||
programmedAmount = insulinDecode(asUINT8(data[1]),asUINT8(data[2]));
|
||||
deliveredAmount = insulinDecode(asUINT8(data[3]),asUINT8(data[4]));
|
||||
unabsorbedInsulinTotal = insulinDecode(asUINT8(data[5]),asUINT8(data[6]));
|
||||
duration = asUINT8(data[7]) * 30;
|
||||
try {
|
||||
timestamp = new PumpTimeStamp(TimeFormat.parse5ByteDate(data, 8));
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
programmedAmount = asUINT8(data[1]) / 10.0f;
|
||||
deliveredAmount = asUINT8(data[2]) / 10.0f;
|
||||
duration = asUINT8(data[3]) * 30;
|
||||
unabsorbedInsulinTotal = 0;
|
||||
try {
|
||||
timestamp = new PumpTimeStamp(TimeFormat.parse5ByteDate(data,4));
|
||||
} catch (org.joda.time.IllegalFieldValueException e) {
|
||||
return false;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
bolusType = (duration > 0) ? "square" : "normal";
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean readFromBundle(Bundle in) {
|
||||
programmedAmount = in.getDouble("programmedAmount",0.0);
|
||||
deliveredAmount = in.getDouble("deliveredAmount",0.0);
|
||||
duration = in.getInt("duration",0);
|
||||
unabsorbedInsulinTotal = in.getDouble("unabsorbedInsulinTotal",0.0);
|
||||
bolusType = in.getString("bolusType","Unset");
|
||||
return super.readFromBundle(in);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToBundle(Bundle in) {
|
||||
super.writeToBundle(in);
|
||||
in.putDouble("programmedAmount",programmedAmount);
|
||||
in.putDouble("deliveredAmount",deliveredAmount);
|
||||
in.putInt("duration",duration);
|
||||
in.putDouble("unabsorbedInsulinTotal",unabsorbedInsulinTotal);
|
||||
in.putString("bolusType",bolusType);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,118 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
import android.os.Bundle;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
public class BolusWizardBolusEstimatePumpEvent extends TimeStampedRecord {
|
||||
private final static String TAG = "BolusWizardBolusEstimatePumpEvent";
|
||||
|
||||
private int carbohydrates;
|
||||
private int bloodGlucose;
|
||||
private double foodEstimate;
|
||||
private double correctionEstimate;
|
||||
private double bolusEstimate;
|
||||
private double unabsorbedInsulinTotal;
|
||||
private int bgTargetLow;
|
||||
private int bgTargetHigh;
|
||||
private int insulinSensitivity;
|
||||
private double carbRatio;
|
||||
|
||||
public BolusWizardBolusEstimatePumpEvent() {
|
||||
correctionEstimate = (double)0.0;
|
||||
bloodGlucose = 0;
|
||||
carbohydrates = 0;
|
||||
carbRatio = 0.0;
|
||||
insulinSensitivity = 0;
|
||||
bgTargetLow = 0;
|
||||
bgTargetHigh = 0;
|
||||
bolusEstimate = 0.0;
|
||||
foodEstimate = 0.0;
|
||||
unabsorbedInsulinTotal = 0.0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getLength() { return PumpModel.isLargerFormat(model) ? 22 : 20;}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Bolus Wizard Est.";
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean readFromBundle(Bundle in) {
|
||||
carbohydrates = in.getInt("carbohydrates",0);
|
||||
bloodGlucose = in.getInt("bloodGlucose",0);
|
||||
foodEstimate = in.getDouble("foodEstimate",0);
|
||||
correctionEstimate = in.getDouble("correctionEstimate",0);
|
||||
bolusEstimate = in.getDouble("bolusEstimate",0);
|
||||
unabsorbedInsulinTotal = in.getDouble("unabsorbedInsulinTotal",0);
|
||||
bgTargetLow = in.getInt("bgTargetLow",0);
|
||||
bgTargetHigh = in.getInt("bgTargetHigh",0);
|
||||
insulinSensitivity = in.getInt("insulinSensitivity",0);
|
||||
carbRatio = in.getDouble("carbRatio",0);
|
||||
return super.readFromBundle(in);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToBundle(Bundle in) {
|
||||
super.writeToBundle(in);
|
||||
in.putInt("carbohydrates",carbohydrates);
|
||||
in.putInt("bloodGlucose",bloodGlucose);
|
||||
in.putDouble("foodEstimate",foodEstimate);
|
||||
in.putDouble("correctionEstimate",correctionEstimate);
|
||||
in.putDouble("bolusEstimate",bolusEstimate);
|
||||
in.putDouble("unabsorbedInsulinTotal",unabsorbedInsulinTotal);
|
||||
in.putInt("bgTargetLow",bgTargetLow);
|
||||
in.putInt("bgTargetHigh",bgTargetHigh);
|
||||
in.putInt("insulinSensitivity",insulinSensitivity);
|
||||
in.putDouble("carbRatio",carbRatio);
|
||||
}
|
||||
|
||||
public double getCorrectionEstimate() { return correctionEstimate; }
|
||||
public long getBG() { return bloodGlucose; }
|
||||
public int getCarbohydrates() { return carbohydrates; }
|
||||
public double getICRatio() { return carbRatio; }
|
||||
public int getInsulinSensitivity() { return insulinSensitivity; }
|
||||
public int getBgTargetLow() { return bgTargetLow; }
|
||||
public int getBgTargetHigh() { return bgTargetHigh; }
|
||||
public double getBolusEstimate() { return bolusEstimate; }
|
||||
public double getFoodEstimate() { return foodEstimate; }
|
||||
public double getUnabsorbedInsulinTotal() { return unabsorbedInsulinTotal; }
|
||||
|
||||
private double insulinDecode(int a, int b) {
|
||||
return ((a << 8) + b) / 40.0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean parseFrom(byte[] data, PumpModel model) {
|
||||
if (!simpleParse(data,2)) {
|
||||
return false;
|
||||
}
|
||||
if (PumpModel.isLargerFormat(model)) {
|
||||
carbohydrates = (asUINT8(data[8]) & 0x0c << 6) + asUINT8(data[7]);
|
||||
bloodGlucose = (asUINT8(data[8]) & 0x03 << 8) + asUINT8(data[1]);
|
||||
foodEstimate = insulinDecode(asUINT8(data[14]), asUINT8(data[15]));
|
||||
correctionEstimate = (double)((asUINT8(data[16])& 0b111000) << 5 + asUINT8(data[13]))/40.0;
|
||||
bolusEstimate = insulinDecode(asUINT8(data[19]), asUINT8(data[20]));
|
||||
unabsorbedInsulinTotal = insulinDecode(asUINT8(data[17]),asUINT8(data[18]));
|
||||
bgTargetLow = asUINT8(data[12]);
|
||||
bgTargetHigh = asUINT8(data[21]);
|
||||
insulinSensitivity = asUINT8(data[11]);
|
||||
carbRatio = (double)(((asUINT8(data[9]) & 0x07) << 8) + asUINT8(data[10]))/40.0;
|
||||
} else {
|
||||
carbohydrates = asUINT8(data[7]);
|
||||
bloodGlucose = ((asUINT8(data[8]) & 0x03) << 8) + asUINT8(data[1]);
|
||||
foodEstimate = (double)(asUINT8(data[13]))/10.0;
|
||||
correctionEstimate = (double)((asUINT8(data[14])<<8) + asUINT8(data[12])) / 10.0;
|
||||
bolusEstimate = (double)(asUINT8(data[18]))/10.0;
|
||||
unabsorbedInsulinTotal = (double)(asUINT8(data[16])) / 10.0;
|
||||
bgTargetLow = asUINT8(data[11]);
|
||||
bgTargetHigh = asUINT8(data[19]);
|
||||
insulinSensitivity = asUINT8(data[10]);
|
||||
carbRatio = (double)asUINT8(data[9]);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
|
@ -1,39 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
import android.os.Bundle;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
public class CalBgForPhPumpEvent extends TimeStampedRecord {
|
||||
private int amount = 0;
|
||||
|
||||
public CalBgForPhPumpEvent() {
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Cal Bg For Ph";
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean parseFrom(byte[] data, PumpModel model) {
|
||||
if (!simpleParse(data,2)) {
|
||||
return false;
|
||||
}
|
||||
amount = ((asUINT8(data[6]) & 0x80) << 1) + asUINT8(data[1]);
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean readFromBundle(Bundle in) {
|
||||
amount = in.getInt("amount",0);
|
||||
return super.readFromBundle(in);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToBundle(Bundle in) {
|
||||
super.writeToBundle(in);
|
||||
in.putInt("amount",amount);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeAlarmClockEnablePumpEvent extends TimeStampedRecord {
|
||||
public ChangeAlarmClockEnablePumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Alarm Clock Enable";
|
||||
}
|
||||
}
|
|
@ -1,10 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
public class ChangeAlarmNotifyModePumpEvent extends TimeStampedRecord {
|
||||
public ChangeAlarmNotifyModePumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Alarm Notify Mode";
|
||||
}
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeAudioBolusPumpEvent extends TimeStampedRecord {
|
||||
public ChangeAudioBolusPumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Audio Bolus";
|
||||
}
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeBGReminderEnablePumpEvent extends TimeStampedRecord {
|
||||
public ChangeBGReminderEnablePumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch BG Rmndr Enable";
|
||||
}
|
||||
}
|
|
@ -1,21 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeBasalProfilePatternPumpEvent extends TimeStampedRecord {
|
||||
public ChangeBasalProfilePatternPumpEvent() {}
|
||||
|
||||
@Override
|
||||
public int getLength() {
|
||||
return 152;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Basal Prof Pat";
|
||||
}
|
||||
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
public class ChangeBasalProfilePumpEvent extends TimeStampedRecord {
|
||||
public ChangeBasalProfilePumpEvent() {
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getLength() { return 152; }
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Basal Profile";
|
||||
}
|
||||
|
||||
}
|
|
@ -1,19 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeBolusReminderEnablePumpEvent extends TimeStampedRecord {
|
||||
public ChangeBolusReminderEnablePumpEvent() {}
|
||||
|
||||
@Override
|
||||
public int getLength() { return 9; }
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Bolus Rmndr Enable";
|
||||
}
|
||||
|
||||
}
|
|
@ -1,19 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeBolusReminderTimePumpEvent extends TimeStampedRecord {
|
||||
public ChangeBolusReminderTimePumpEvent(){}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Bolus Rmndr Time";
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getLength() { return 9; }
|
||||
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeBolusScrollStepSizePumpEvent extends TimeStampedRecord {
|
||||
public ChangeBolusScrollStepSizePumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Bolus Scroll SS";
|
||||
}
|
||||
}
|
|
@ -1,21 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
|
||||
import com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpModel;
|
||||
|
||||
public class ChangeBolusWizardSetupPumpEvent extends TimeStampedRecord {
|
||||
|
||||
public ChangeBolusWizardSetupPumpEvent() {
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getLength() {
|
||||
return 144;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Bolus Wizard Setup";
|
||||
}
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeCaptureEventEnablePumpEvent extends TimeStampedRecord {
|
||||
public ChangeCaptureEventEnablePumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Capture Event Ena";
|
||||
}
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeCarbUnitsPumpEvent extends TimeStampedRecord {
|
||||
public ChangeCarbUnitsPumpEvent() {}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Carb Units";
|
||||
}
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
package com.gxwtech.roundtrip2.RoundtripService.medtronic.PumpData.records;
|
||||
|
||||
/**
|
||||
* Created by geoff on 6/5/16.
|
||||
*/
|
||||
public class ChangeChildBlockEnablePumpEvent extends TimeStampedRecord {
|
||||
public ChangeChildBlockEnablePumpEvent(){}
|
||||
|
||||
@Override
|
||||
public String getShortTypeName() {
|
||||
return "Ch Child Block Ena";
|
||||
}
|
||||
}
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue