confirm/deny commands
This commit is contained in:
parent
feed6a2f63
commit
842f196ae8
19 changed files with 212 additions and 94 deletions
|
@ -11,18 +11,23 @@ import info.nightscout.androidaps.plugins.common.ManufacturerType
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType
|
||||||
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType
|
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.common.definition.OmnipodCommandType
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.common.queue.command.*
|
import info.nightscout.androidaps.plugins.pump.omnipod.common.queue.command.*
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.OmnipodDashManager
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.OmnipodDashManager
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.ActivationProgress
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.ActivationProgress
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.BeepType
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.BeepType
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.ResponseType
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.ResponseType
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.state.OmnipodDashPodStateManager
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.state.OmnipodDashPodStateManager
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.DashHistory
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.BolusRecord
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.BolusType
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.ui.OmnipodDashOverviewFragment
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.ui.OmnipodDashOverviewFragment
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.util.mapProfileToBasalProgram
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.util.mapProfileToBasalProgram
|
||||||
import info.nightscout.androidaps.queue.commands.CustomCommand
|
import info.nightscout.androidaps.queue.commands.CustomCommand
|
||||||
import info.nightscout.androidaps.utils.TimeChangeType
|
import info.nightscout.androidaps.utils.TimeChangeType
|
||||||
import info.nightscout.androidaps.utils.resources.ResourceHelper
|
import info.nightscout.androidaps.utils.resources.ResourceHelper
|
||||||
import info.nightscout.androidaps.utils.sharedPreferences.SP
|
import info.nightscout.androidaps.utils.sharedPreferences.SP
|
||||||
|
import io.reactivex.Observable
|
||||||
import io.reactivex.Single
|
import io.reactivex.Single
|
||||||
import io.reactivex.rxkotlin.blockingSubscribeBy
|
import io.reactivex.rxkotlin.blockingSubscribeBy
|
||||||
import io.reactivex.rxkotlin.subscribeBy
|
import io.reactivex.rxkotlin.subscribeBy
|
||||||
|
@ -37,6 +42,7 @@ class OmnipodDashPumpPlugin @Inject constructor(
|
||||||
private val podStateManager: OmnipodDashPodStateManager,
|
private val podStateManager: OmnipodDashPodStateManager,
|
||||||
private val sp: SP,
|
private val sp: SP,
|
||||||
private val profileFunction: ProfileFunction,
|
private val profileFunction: ProfileFunction,
|
||||||
|
private val history: DashHistory,
|
||||||
injector: HasAndroidInjector,
|
injector: HasAndroidInjector,
|
||||||
aapsLogger: AAPSLogger,
|
aapsLogger: AAPSLogger,
|
||||||
resourceHelper: ResourceHelper,
|
resourceHelper: ResourceHelper,
|
||||||
|
@ -147,7 +153,6 @@ class OmnipodDashPumpPlugin @Inject constructor(
|
||||||
it == mapProfileToBasalProgram(profile)
|
it == mapProfileToBasalProgram(profile)
|
||||||
} ?: true
|
} ?: true
|
||||||
|
|
||||||
|
|
||||||
override fun lastDataTime(): Long {
|
override fun lastDataTime(): Long {
|
||||||
return podStateManager.lastConnection
|
return podStateManager.lastConnection
|
||||||
}
|
}
|
||||||
|
@ -172,7 +177,6 @@ class OmnipodDashPumpPlugin @Inject constructor(
|
||||||
get() = 0
|
get() = 0
|
||||||
|
|
||||||
override fun deliverTreatment(detailedBolusInfo: DetailedBolusInfo): PumpEnactResult {
|
override fun deliverTreatment(detailedBolusInfo: DetailedBolusInfo): PumpEnactResult {
|
||||||
// TODO history
|
|
||||||
// TODO update Treatments (?)
|
// TODO update Treatments (?)
|
||||||
// TODO bolus progress
|
// TODO bolus progress
|
||||||
// TODO report actual delivered amount after Pod Alarm and bolus cancellation
|
// TODO report actual delivered amount after Pod Alarm and bolus cancellation
|
||||||
|
@ -180,29 +184,44 @@ class OmnipodDashPumpPlugin @Inject constructor(
|
||||||
return Single.create<PumpEnactResult> { source ->
|
return Single.create<PumpEnactResult> { source ->
|
||||||
val bolusBeeps = sp.getBoolean(R.string.key_omnipod_common_bolus_beeps_enabled, false)
|
val bolusBeeps = sp.getBoolean(R.string.key_omnipod_common_bolus_beeps_enabled, false)
|
||||||
|
|
||||||
omnipodManager.bolus(
|
Observable.concat(
|
||||||
detailedBolusInfo.insulin,
|
history.createRecord(
|
||||||
bolusBeeps,
|
commandType = OmnipodCommandType.SET_BOLUS,
|
||||||
bolusBeeps
|
bolusRecord = BolusRecord(
|
||||||
).subscribeBy(
|
detailedBolusInfo.insulin,
|
||||||
onNext = { podEvent ->
|
if (detailedBolusInfo.isSMB) BolusType.SMB else BolusType.DEFAULT
|
||||||
aapsLogger.debug(
|
),
|
||||||
LTag.PUMP,
|
).flatMapObservable {
|
||||||
"Received PodEvent in deliverTreatment: $podEvent"
|
recordId ->
|
||||||
)
|
podStateManager.createActiveCommand(recordId).toObservable()
|
||||||
},
|
},
|
||||||
onError = { throwable ->
|
omnipodManager.bolus(
|
||||||
aapsLogger.error(LTag.PUMP, "Error in deliverTreatment", throwable)
|
detailedBolusInfo.insulin,
|
||||||
source.onSuccess(PumpEnactResult(injector).success(false).enacted(false).comment(throwable.message))
|
bolusBeeps,
|
||||||
},
|
bolusBeeps
|
||||||
onComplete = {
|
),
|
||||||
aapsLogger.debug("deliverTreatment completed")
|
history.updateFromState(podStateManager).toObservable(),
|
||||||
source.onSuccess(
|
podStateManager.updateActiveCommand().toObservable(),
|
||||||
PumpEnactResult(injector).success(true).enacted(true).bolusDelivered(detailedBolusInfo.insulin)
|
|
||||||
.carbsDelivered(detailedBolusInfo.carbs)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
|
.subscribeBy(
|
||||||
|
onNext = { podEvent ->
|
||||||
|
aapsLogger.debug(
|
||||||
|
LTag.PUMP,
|
||||||
|
"Received PodEvent in deliverTreatment: $podEvent"
|
||||||
|
)
|
||||||
|
},
|
||||||
|
onError = { throwable ->
|
||||||
|
aapsLogger.error(LTag.PUMP, "Error in deliverTreatment", throwable)
|
||||||
|
source.onSuccess(PumpEnactResult(injector).success(false).enacted(false).comment(throwable.message))
|
||||||
|
},
|
||||||
|
onComplete = {
|
||||||
|
aapsLogger.debug("deliverTreatment completed")
|
||||||
|
source.onSuccess(
|
||||||
|
PumpEnactResult(injector).success(true).enacted(true).bolusDelivered(detailedBolusInfo.insulin)
|
||||||
|
.carbsDelivered(detailedBolusInfo.carbs)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
)
|
||||||
}.blockingGet()
|
}.blockingGet()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver
|
package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver
|
||||||
|
|
||||||
|
import android.os.SystemClock
|
||||||
import info.nightscout.androidaps.logging.AAPSLogger
|
import info.nightscout.androidaps.logging.AAPSLogger
|
||||||
import info.nightscout.androidaps.logging.LTag
|
import info.nightscout.androidaps.logging.LTag
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.OmnipodDashBleManager
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.OmnipodDashBleManager
|
||||||
|
@ -643,6 +644,11 @@ class OmnipodDashManagerImpl @Inject constructor(
|
||||||
}
|
}
|
||||||
|
|
||||||
is PodEvent.CommandSent -> {
|
is PodEvent.CommandSent -> {
|
||||||
|
podStateManager.activeCommand?.let {
|
||||||
|
if (it.sequence == event.command.sequenceNumber) {
|
||||||
|
it.sentRealtime = SystemClock.elapsedRealtime()
|
||||||
|
}
|
||||||
|
}
|
||||||
podStateManager.increaseMessageSequenceNumber()
|
podStateManager.increaseMessageSequenceNumber()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,6 @@ package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm
|
||||||
import info.nightscout.androidaps.utils.extensions.toHex
|
import info.nightscout.androidaps.utils.extensions.toHex
|
||||||
import java.nio.ByteBuffer
|
import java.nio.ByteBuffer
|
||||||
|
|
||||||
|
|
||||||
data class Id(val address: ByteArray) {
|
data class Id(val address: ByteArray) {
|
||||||
init {
|
init {
|
||||||
require(address.size == 4)
|
require(address.size == 4)
|
||||||
|
|
|
@ -83,7 +83,7 @@ class OmnipodDashBleManagerImpl @Inject constructor(
|
||||||
val conn = assertConnected()
|
val conn = assertConnected()
|
||||||
return conn.session
|
return conn.session
|
||||||
?: throw NotConnectedException("Missing session")
|
?: throw NotConnectedException("Missing session")
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun getStatus(): ConnectionStatus {
|
override fun getStatus(): ConnectionStatus {
|
||||||
// TODO is this used?
|
// TODO is this used?
|
||||||
|
|
|
@ -42,7 +42,7 @@ class PayloadJoiner(private val firstPacket: ByteArray) {
|
||||||
oneExtraPacket = lastPacket.oneExtraPacket
|
oneExtraPacket = lastPacket.oneExtraPacket
|
||||||
}
|
}
|
||||||
|
|
||||||
idx == fullFragments+1 && oneExtraPacket -> {
|
idx == fullFragments + 1 && oneExtraPacket -> {
|
||||||
fragments.add(LastOptionalPlusOneBlePacket.parse(packet))
|
fragments.add(LastOptionalPlusOneBlePacket.parse(packet))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,15 +2,12 @@ package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.pair
|
||||||
|
|
||||||
import info.nightscout.androidaps.logging.AAPSLogger
|
import info.nightscout.androidaps.logging.AAPSLogger
|
||||||
import info.nightscout.androidaps.logging.LTag
|
import info.nightscout.androidaps.logging.LTag
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Id
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Ids
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Ids
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.MessageIOException
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.MessageIOException
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.PairingException
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.PairingException
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.MessageIO
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.MessageIO
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.MessagePacket
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.MessagePacket
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.MessageSendErrorSending
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.MessageSendSuccess
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.MessageSendSuccess
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.StringLengthPrefixEncoding
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.StringLengthPrefixEncoding.Companion.parseKeys
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.message.StringLengthPrefixEncoding.Companion.parseKeys
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.util.RandomByteGenerator
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.util.RandomByteGenerator
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.util.X25519KeyGenerator
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.util.X25519KeyGenerator
|
||||||
|
@ -35,7 +32,7 @@ internal class LTKExchanger(
|
||||||
keys = arrayOf(SP1, SP2),
|
keys = arrayOf(SP1, SP2),
|
||||||
payloads = arrayOf(ids.podId.address, sp2())
|
payloads = arrayOf(ids.podId.address, sp2())
|
||||||
)
|
)
|
||||||
throwOnSendError(sp1sp2.messagePacket, SP1+SP2)
|
throwOnSendError(sp1sp2.messagePacket, SP1 + SP2)
|
||||||
|
|
||||||
seq++
|
seq++
|
||||||
val sps1 = PairMessage(
|
val sps1 = PairMessage(
|
||||||
|
@ -67,16 +64,16 @@ internal class LTKExchanger(
|
||||||
|
|
||||||
seq++
|
seq++
|
||||||
// send SP0GP0
|
// send SP0GP0
|
||||||
val sp0gp0 = PairMessage (
|
val sp0gp0 = PairMessage(
|
||||||
sequenceNumber = seq,
|
sequenceNumber = seq,
|
||||||
source = ids.myId,
|
source = ids.myId,
|
||||||
destination = podAddress,
|
destination = podAddress,
|
||||||
keys = arrayOf(SP0GP0),
|
keys = arrayOf(SP0GP0),
|
||||||
payloads = arrayOf(ByteArray(0))
|
payloads = arrayOf(ByteArray(0))
|
||||||
)
|
)
|
||||||
val result = msgIO.sendMessage(sp0gp0.messagePacket)
|
val result = msgIO.sendMessage(sp0gp0.messagePacket)
|
||||||
if (result !is MessageSendSuccess) {
|
if (result !is MessageSendSuccess) {
|
||||||
aapsLogger.warn(LTag.PUMPBTCOMM,"Error sending SP0GP0: $result")
|
aapsLogger.warn(LTag.PUMPBTCOMM, "Error sending SP0GP0: $result")
|
||||||
}
|
}
|
||||||
|
|
||||||
msgIO.receiveMessage()
|
msgIO.receiveMessage()
|
||||||
|
@ -97,7 +94,6 @@ internal class LTKExchanger(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private fun processSps1FromPod(msg: MessagePacket) {
|
private fun processSps1FromPod(msg: MessagePacket) {
|
||||||
aapsLogger.debug(LTag.PUMPBTCOMM, "Received SPS1 from pod: ${msg.payload.toHex()}")
|
aapsLogger.debug(LTag.PUMPBTCOMM, "Received SPS1 from pod: ${msg.payload.toHex()}")
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,6 @@ package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.session
|
||||||
|
|
||||||
import info.nightscout.androidaps.logging.AAPSLogger
|
import info.nightscout.androidaps.logging.AAPSLogger
|
||||||
import info.nightscout.androidaps.logging.LTag
|
import info.nightscout.androidaps.logging.LTag
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Id
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Ids
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Ids
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.endecrypt.EnDecrypt
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.endecrypt.EnDecrypt
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.CouldNotParseResponseException
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.CouldNotParseResponseException
|
||||||
|
|
|
@ -2,7 +2,6 @@ package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.session
|
||||||
|
|
||||||
import info.nightscout.androidaps.logging.AAPSLogger
|
import info.nightscout.androidaps.logging.AAPSLogger
|
||||||
import info.nightscout.androidaps.logging.LTag
|
import info.nightscout.androidaps.logging.LTag
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Id
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Ids
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Ids
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.endecrypt.Nonce
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.endecrypt.Nonce
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.SessionEstablishmentException
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.exceptions.SessionEstablishmentException
|
||||||
|
|
|
@ -65,4 +65,8 @@ sealed class PodEvent {
|
||||||
return "ResponseReceived(command=$command, response=$response)"
|
return "ResponseReceived(command=$command, response=$response)"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
data class CommandConfirmed(val historyId: String) : PodEvent()
|
||||||
|
|
||||||
|
data class CommandDenied(val historyId: String) : PodEvent()
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,7 +31,7 @@ class ProgramBasalCommand private constructor(
|
||||||
private val delayUntilNextTenthPulseInUsec: Int
|
private val delayUntilNextTenthPulseInUsec: Int
|
||||||
val length: Short
|
val length: Short
|
||||||
get() = (insulinProgramElements.size * 6 + 10).toShort()
|
get() = (insulinProgramElements.size * 6 + 10).toShort()
|
||||||
val bodyLength: Byte
|
private val bodyLength: Byte
|
||||||
get() = (insulinProgramElements.size * 6 + 8).toByte()
|
get() = (insulinProgramElements.size * 6 + 8).toByte()
|
||||||
|
|
||||||
override val encoded: ByteArray
|
override val encoded: ByteArray
|
||||||
|
|
|
@ -6,4 +6,5 @@ import java.io.Serializable
|
||||||
interface Command : Encodable, Serializable {
|
interface Command : Encodable, Serializable {
|
||||||
|
|
||||||
val commandType: CommandType
|
val commandType: CommandType
|
||||||
|
val sequenceNumber: Short
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@ import java.nio.ByteBuffer
|
||||||
abstract class HeaderEnabledCommand protected constructor(
|
abstract class HeaderEnabledCommand protected constructor(
|
||||||
override val commandType: CommandType,
|
override val commandType: CommandType,
|
||||||
protected val uniqueId: Int,
|
protected val uniqueId: Int,
|
||||||
protected val sequenceNumber: Short,
|
override val sequenceNumber: Short,
|
||||||
protected val multiCommandFlag: Boolean
|
protected val multiCommandFlag: Boolean
|
||||||
) : Command {
|
) : Command {
|
||||||
|
|
||||||
|
|
|
@ -2,11 +2,14 @@ package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.state
|
||||||
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Id
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Id
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.pair.PairResult
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.pair.PairResult
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.event.PodEvent
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.*
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.*
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.AlarmStatusResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.AlarmStatusResponse
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.DefaultStatusResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.DefaultStatusResponse
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.SetUniqueIdResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.SetUniqueIdResponse
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.VersionResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.VersionResponse
|
||||||
|
import io.reactivex.Completable
|
||||||
|
import io.reactivex.Maybe
|
||||||
import java.io.Serializable
|
import java.io.Serializable
|
||||||
import java.util.*
|
import java.util.*
|
||||||
|
|
||||||
|
@ -18,7 +21,10 @@ interface OmnipodDashPodStateManager {
|
||||||
val isSuspended: Boolean
|
val isSuspended: Boolean
|
||||||
val isPodRunning: Boolean
|
val isPodRunning: Boolean
|
||||||
var lastConnection: Long
|
var lastConnection: Long
|
||||||
val lastUpdated: Long
|
|
||||||
|
val lastUpdatedSystem: Long // System.currentTimeMillis()
|
||||||
|
// TODO: set lastUpdatedRealtime to 0 on boot
|
||||||
|
val lastStatusResponseReceived: Long
|
||||||
|
|
||||||
val messageSequenceNumber: Short
|
val messageSequenceNumber: Short
|
||||||
val sequenceNumberOfLastProgrammingCommand: Short?
|
val sequenceNumberOfLastProgrammingCommand: Short?
|
||||||
|
@ -48,6 +54,7 @@ interface OmnipodDashPodStateManager {
|
||||||
val tempBasal: TempBasal?
|
val tempBasal: TempBasal?
|
||||||
val tempBasalActive: Boolean
|
val tempBasalActive: Boolean
|
||||||
var basalProgram: BasalProgram?
|
var basalProgram: BasalProgram?
|
||||||
|
val activeCommand: ActiveCommand?
|
||||||
|
|
||||||
fun increaseMessageSequenceNumber()
|
fun increaseMessageSequenceNumber()
|
||||||
fun increaseEapAkaSequenceNumber(): ByteArray
|
fun increaseEapAkaSequenceNumber(): ByteArray
|
||||||
|
@ -59,5 +66,15 @@ interface OmnipodDashPodStateManager {
|
||||||
fun updateFromPairing(uniqueId: Id, pairResult: PairResult)
|
fun updateFromPairing(uniqueId: Id, pairResult: PairResult)
|
||||||
fun reset()
|
fun reset()
|
||||||
|
|
||||||
|
fun createActiveCommand(historyId: String): Completable
|
||||||
|
fun updateActiveCommand(): Maybe<PodEvent>
|
||||||
|
|
||||||
|
data class ActiveCommand(
|
||||||
|
val sequence: Short,
|
||||||
|
val createdRealtime: Long,
|
||||||
|
var sentRealtime: Long = 0,
|
||||||
|
val historyId: String
|
||||||
|
)
|
||||||
|
// TODO: set created to "now" on boot
|
||||||
data class TempBasal(val startTime: Long, val rate: Double, val durationInMinutes: Short) : Serializable
|
data class TempBasal(val startTime: Long, val rate: Double, val durationInMinutes: Short) : Serializable
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.state
|
package info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.state
|
||||||
|
|
||||||
|
import android.os.SystemClock
|
||||||
import com.google.gson.Gson
|
import com.google.gson.Gson
|
||||||
import info.nightscout.androidaps.logging.AAPSLogger
|
import info.nightscout.androidaps.logging.AAPSLogger
|
||||||
import info.nightscout.androidaps.logging.LTag
|
import info.nightscout.androidaps.logging.LTag
|
||||||
|
@ -9,12 +10,15 @@ import info.nightscout.androidaps.plugins.pump.omnipod.dash.R
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Id
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.Id
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.pair.PairResult
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.pair.PairResult
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.session.EapSqn
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.comm.session.EapSqn
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.event.PodEvent
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.*
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.definition.*
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.AlarmStatusResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.AlarmStatusResponse
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.DefaultStatusResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.DefaultStatusResponse
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.SetUniqueIdResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.SetUniqueIdResponse
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.VersionResponse
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.response.VersionResponse
|
||||||
import info.nightscout.androidaps.utils.sharedPreferences.SP
|
import info.nightscout.androidaps.utils.sharedPreferences.SP
|
||||||
|
import io.reactivex.Completable
|
||||||
|
import io.reactivex.Maybe
|
||||||
import java.io.Serializable
|
import java.io.Serializable
|
||||||
import java.util.*
|
import java.util.*
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
|
@ -60,8 +64,8 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
store()
|
store()
|
||||||
}
|
}
|
||||||
|
|
||||||
override val lastUpdated: Long
|
override val lastUpdatedSystem: Long
|
||||||
get() = podState.lastUpdated
|
get() = podState.lastUpdatedSystem
|
||||||
|
|
||||||
override val messageSequenceNumber: Short
|
override val messageSequenceNumber: Short
|
||||||
get() = podState.messageSequenceNumber
|
get() = podState.messageSequenceNumber
|
||||||
|
@ -152,6 +156,9 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
store()
|
store()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override val lastStatusResponseReceived: Long
|
||||||
|
get() = podState.lastStatusResponseReceived
|
||||||
|
|
||||||
override fun increaseMessageSequenceNumber() {
|
override fun increaseMessageSequenceNumber() {
|
||||||
podState.messageSequenceNumber = ((podState.messageSequenceNumber.toInt() + 1) and 0x0f).toShort()
|
podState.messageSequenceNumber = ((podState.messageSequenceNumber.toInt() + 1) and 0x0f).toShort()
|
||||||
store()
|
store()
|
||||||
|
@ -171,6 +178,41 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
store()
|
store()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override val activeCommand: OmnipodDashPodStateManager.ActiveCommand?
|
||||||
|
get() = podState.activeCommand
|
||||||
|
|
||||||
|
@Synchronized
|
||||||
|
override fun createActiveCommand(historyId: String): Completable {
|
||||||
|
return if (activeCommand == null) {
|
||||||
|
podState.activeCommand = OmnipodDashPodStateManager.ActiveCommand(
|
||||||
|
podState.messageSequenceNumber,
|
||||||
|
createdRealtime = SystemClock.elapsedRealtime(),
|
||||||
|
historyId = historyId
|
||||||
|
)
|
||||||
|
Completable.complete()
|
||||||
|
} else {
|
||||||
|
Completable.error(
|
||||||
|
java.lang.IllegalStateException(
|
||||||
|
"Trying to send a command " +
|
||||||
|
"and the last command was not confirmed"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Synchronized
|
||||||
|
override fun updateActiveCommand(): Maybe<PodEvent> {
|
||||||
|
return podState.activeCommand?.run {
|
||||||
|
if (createdRealtime >= lastStatusResponseReceived)
|
||||||
|
Maybe.empty()
|
||||||
|
else if (sequenceNumberOfLastProgrammingCommand == sequence)
|
||||||
|
Maybe.just(PodEvent.CommandConfirmed(historyId))
|
||||||
|
else
|
||||||
|
Maybe.just(PodEvent.CommandDenied(historyId))
|
||||||
|
}
|
||||||
|
?: Maybe.empty() // no active programming command
|
||||||
|
}
|
||||||
|
|
||||||
override fun increaseEapAkaSequenceNumber(): ByteArray {
|
override fun increaseEapAkaSequenceNumber(): ByteArray {
|
||||||
podState.eapAkaSequenceNumber++
|
podState.eapAkaSequenceNumber++
|
||||||
return EapSqn(podState.eapAkaSequenceNumber).value
|
return EapSqn(podState.eapAkaSequenceNumber).value
|
||||||
|
@ -191,7 +233,9 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
podState.minutesSinceActivation = response.minutesSinceActivation
|
podState.minutesSinceActivation = response.minutesSinceActivation
|
||||||
podState.activeAlerts = response.activeAlerts
|
podState.activeAlerts = response.activeAlerts
|
||||||
|
|
||||||
podState.lastUpdated = System.currentTimeMillis()
|
podState.lastUpdatedSystem = System.currentTimeMillis()
|
||||||
|
podState.lastStatusResponseReceived = SystemClock.elapsedRealtime()
|
||||||
|
|
||||||
store()
|
store()
|
||||||
rxBus.send(EventOmnipodDashPumpValuesChanged())
|
rxBus.send(EventOmnipodDashPumpValuesChanged())
|
||||||
}
|
}
|
||||||
|
@ -211,7 +255,8 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
podState.lotNumber = response.lotNumber
|
podState.lotNumber = response.lotNumber
|
||||||
podState.podSequenceNumber = response.podSequenceNumber
|
podState.podSequenceNumber = response.podSequenceNumber
|
||||||
|
|
||||||
podState.lastUpdated = System.currentTimeMillis()
|
podState.lastUpdatedSystem = System.currentTimeMillis()
|
||||||
|
|
||||||
store()
|
store()
|
||||||
rxBus.send(EventOmnipodDashPumpValuesChanged())
|
rxBus.send(EventOmnipodDashPumpValuesChanged())
|
||||||
}
|
}
|
||||||
|
@ -237,7 +282,8 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
podState.podSequenceNumber = response.podSequenceNumber
|
podState.podSequenceNumber = response.podSequenceNumber
|
||||||
podState.uniqueId = response.uniqueIdReceivedInCommand
|
podState.uniqueId = response.uniqueIdReceivedInCommand
|
||||||
|
|
||||||
podState.lastUpdated = System.currentTimeMillis()
|
podState.lastUpdatedSystem = System.currentTimeMillis()
|
||||||
|
|
||||||
store()
|
store()
|
||||||
rxBus.send(EventOmnipodDashPumpValuesChanged())
|
rxBus.send(EventOmnipodDashPumpValuesChanged())
|
||||||
}
|
}
|
||||||
|
@ -293,7 +339,8 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
|
|
||||||
var activationProgress: ActivationProgress = ActivationProgress.NOT_STARTED
|
var activationProgress: ActivationProgress = ActivationProgress.NOT_STARTED
|
||||||
var lastConnection: Long = 0
|
var lastConnection: Long = 0
|
||||||
var lastUpdated: Long = 0
|
var lastUpdatedSystem: Long = 0
|
||||||
|
var lastStatusResponseReceived: Long = 0
|
||||||
|
|
||||||
var messageSequenceNumber: Short = 0
|
var messageSequenceNumber: Short = 0
|
||||||
var sequenceNumberOfLastProgrammingCommand: Short? = null
|
var sequenceNumberOfLastProgrammingCommand: Short? = null
|
||||||
|
@ -322,5 +369,6 @@ class OmnipodDashPodStateManagerImpl @Inject constructor(
|
||||||
|
|
||||||
var basalProgram: BasalProgram? = null
|
var basalProgram: BasalProgram? = null
|
||||||
var tempBasal: OmnipodDashPodStateManager.TempBasal? = null
|
var tempBasal: OmnipodDashPodStateManager.TempBasal? = null
|
||||||
|
var activeCommand: OmnipodDashPodStateManager.ActiveCommand? = null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ import com.github.guepardoapps.kulid.ULID
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.common.definition.OmnipodCommandType
|
import info.nightscout.androidaps.plugins.pump.omnipod.common.definition.OmnipodCommandType
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.common.definition.OmnipodCommandType.SET_BOLUS
|
import info.nightscout.androidaps.plugins.pump.omnipod.common.definition.OmnipodCommandType.SET_BOLUS
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.common.definition.OmnipodCommandType.SET_TEMPORARY_BASAL
|
import info.nightscout.androidaps.plugins.pump.omnipod.common.definition.OmnipodCommandType.SET_TEMPORARY_BASAL
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.driver.pod.state.OmnipodDashPodStateManager
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.BolusRecord
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.BolusRecord
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.HistoryRecord
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.HistoryRecord
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.InitialResult
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.InitialResult
|
||||||
|
@ -22,13 +23,13 @@ class DashHistory @Inject constructor(
|
||||||
private val historyMapper: HistoryMapper
|
private val historyMapper: HistoryMapper
|
||||||
) {
|
) {
|
||||||
|
|
||||||
fun markSuccess(id: String, date: Long): Completable = dao.markResolved(
|
private fun markSuccess(id: String): Completable = dao.markResolved(
|
||||||
id,
|
id,
|
||||||
ResolvedResult.SUCCESS,
|
ResolvedResult.SUCCESS,
|
||||||
currentTimeMillis()
|
currentTimeMillis()
|
||||||
)
|
)
|
||||||
|
|
||||||
fun markFailure(id: String, date: Long): Completable = dao.markResolved(
|
private fun markFailure(id: String): Completable = dao.markResolved(
|
||||||
id,
|
id,
|
||||||
ResolvedResult.FAILURE,
|
ResolvedResult.FAILURE,
|
||||||
currentTimeMillis()
|
currentTimeMillis()
|
||||||
|
@ -37,8 +38,8 @@ class DashHistory @Inject constructor(
|
||||||
@Suppress("ReturnCount")
|
@Suppress("ReturnCount")
|
||||||
fun createRecord(
|
fun createRecord(
|
||||||
commandType: OmnipodCommandType,
|
commandType: OmnipodCommandType,
|
||||||
date: Long,
|
date: Long = System.currentTimeMillis(),
|
||||||
initialResult: InitialResult = InitialResult.UNCONFIRMED,
|
initialResult: InitialResult = InitialResult.CREATED,
|
||||||
tempBasalRecord: TempBasalRecord? = null,
|
tempBasalRecord: TempBasalRecord? = null,
|
||||||
bolusRecord: BolusRecord? = null,
|
bolusRecord: BolusRecord? = null,
|
||||||
resolveResult: ResolvedResult? = null,
|
resolveResult: ResolvedResult? = null,
|
||||||
|
@ -72,4 +73,29 @@ class DashHistory @Inject constructor(
|
||||||
dao.all().map { list -> list.map(historyMapper::entityToDomain) }
|
dao.all().map { list -> list.map(historyMapper::entityToDomain) }
|
||||||
|
|
||||||
fun getRecordsAfter(time: Long): Single<List<HistoryRecordEntity>> = dao.allSince(time)
|
fun getRecordsAfter(time: Long): Single<List<HistoryRecordEntity>> = dao.allSince(time)
|
||||||
|
|
||||||
|
fun updateFromState(podState: OmnipodDashPodStateManager): Completable {
|
||||||
|
return podState.activeCommand?.run {
|
||||||
|
when {
|
||||||
|
|
||||||
|
createdRealtime <= podState.lastStatusResponseReceived &&
|
||||||
|
sequence == podState.sequenceNumberOfLastProgrammingCommand ->
|
||||||
|
dao.setInitialResult(historyId, InitialResult.SENT)
|
||||||
|
.andThen(markSuccess(historyId))
|
||||||
|
|
||||||
|
createdRealtime <= podState.lastStatusResponseReceived &&
|
||||||
|
sequence != podState.sequenceNumberOfLastProgrammingCommand ->
|
||||||
|
markFailure(historyId)
|
||||||
|
|
||||||
|
// no response received after this point
|
||||||
|
createdRealtime <= sentRealtime ->
|
||||||
|
dao.setInitialResult(historyId, InitialResult.SENT)
|
||||||
|
|
||||||
|
createdRealtime > sentRealtime ->
|
||||||
|
dao.setInitialResult(historyId, InitialResult.FAILURE_SENDING)
|
||||||
|
|
||||||
|
else -> Completable.error(IllegalStateException("This can't happen. Could not update history"))
|
||||||
|
}
|
||||||
|
} ?: Completable.complete() // no active programming command
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
package info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data
|
package info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data
|
||||||
|
|
||||||
enum class InitialResult {
|
enum class InitialResult {
|
||||||
SUCCESS, FAILURE, UNCONFIRMED
|
CREATED, FAILURE_SENDING, UNCONFIRMED, SENT
|
||||||
}
|
}
|
||||||
|
|
||||||
enum class ResolvedResult {
|
enum class ResolvedResult {
|
||||||
|
|
|
@ -5,6 +5,7 @@ import androidx.room.Delete
|
||||||
import androidx.room.Insert
|
import androidx.room.Insert
|
||||||
import androidx.room.OnConflictStrategy
|
import androidx.room.OnConflictStrategy
|
||||||
import androidx.room.Query
|
import androidx.room.Query
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.InitialResult
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.ResolvedResult
|
import info.nightscout.androidaps.plugins.pump.omnipod.dash.history.data.ResolvedResult
|
||||||
import io.reactivex.Completable
|
import io.reactivex.Completable
|
||||||
import io.reactivex.Single
|
import io.reactivex.Single
|
||||||
|
@ -32,4 +33,7 @@ abstract class HistoryRecordDao {
|
||||||
|
|
||||||
@Query("UPDATE historyrecords SET resolvedResult = :resolvedResult, resolvedAt = :resolvedAt WHERE id = :id ")
|
@Query("UPDATE historyrecords SET resolvedResult = :resolvedResult, resolvedAt = :resolvedAt WHERE id = :id ")
|
||||||
abstract fun markResolved(id: String, resolvedResult: ResolvedResult, resolvedAt: Long): Completable
|
abstract fun markResolved(id: String, resolvedResult: ResolvedResult, resolvedAt: Long): Completable
|
||||||
|
|
||||||
|
@Query("UPDATE historyrecords SET initialResult = :initialResult WHERE id = :id ")
|
||||||
|
abstract fun setInitialResult(id: String, initialResult: InitialResult): Completable
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,25 +28,25 @@ class DefaultStatusResponseTest {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* response (hex) 08202EAA0C0A1D1905281000004387D3039A
|
* response (hex) 08202EAA0C0A1D1905281000004387D3039A
|
||||||
Status response: 29
|
Status response: 29
|
||||||
Pod status: RUNNING_BELOW_MIN_VOLUME
|
Pod status: RUNNING_BELOW_MIN_VOLUME
|
||||||
Basal active: true
|
Basal active: true
|
||||||
Temp Basal active: false
|
Temp Basal active: false
|
||||||
Immediate bolus active: false
|
Immediate bolus active: false
|
||||||
Extended bolus active: false
|
Extended bolus active: false
|
||||||
Bolus pulses remaining: 0
|
Bolus pulses remaining: 0
|
||||||
sequence number of last programing command: 2
|
sequence number of last programing command: 2
|
||||||
Total full pulses delivered: 2640
|
Total full pulses delivered: 2640
|
||||||
Full reservoir pulses remaining: 979
|
Full reservoir pulses remaining: 979
|
||||||
Time since activation: 4321
|
Time since activation: 4321
|
||||||
Alert 1 is InActive
|
Alert 1 is InActive
|
||||||
Alert 2 is InActive
|
Alert 2 is InActive
|
||||||
Alert 3 is InActive
|
Alert 3 is InActive
|
||||||
Alert 4 is InActive
|
Alert 4 is InActive
|
||||||
Alert 5 is InActive
|
Alert 5 is InActive
|
||||||
Alert 6 is InActive
|
Alert 6 is InActive
|
||||||
Alert 7 is InActive
|
Alert 7 is InActive
|
||||||
Occlusion alert active false
|
Occlusion alert active false
|
||||||
*/
|
*/
|
||||||
@Test @Throws(DecoderException::class) fun testValidResponseBelowMin() {
|
@Test @Throws(DecoderException::class) fun testValidResponseBelowMin() {
|
||||||
val encoded = Hex.decodeHex("1D1905281000004387D3039A")
|
val encoded = Hex.decodeHex("1D1905281000004387D3039A")
|
||||||
|
@ -67,25 +67,25 @@ class DefaultStatusResponseTest {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* response (hex) 08202EAA080A1D180519C00E0039A7FF8085
|
* response (hex) 08202EAA080A1D180519C00E0039A7FF8085
|
||||||
Status response: 29
|
Status response: 29
|
||||||
Pod status: RUNNING_ABOVE_MIN_VOLUME
|
Pod status: RUNNING_ABOVE_MIN_VOLUME
|
||||||
Basal active: true
|
Basal active: true
|
||||||
Temp Basal active: false
|
Temp Basal active: false
|
||||||
Immediate bolus active: false
|
Immediate bolus active: false
|
||||||
Extended bolus active: false
|
Extended bolus active: false
|
||||||
Bolus pulses remaining: 14
|
Bolus pulses remaining: 14
|
||||||
sequence number of last programing command: 8
|
sequence number of last programing command: 8
|
||||||
Total full pulses delivered: 2611
|
Total full pulses delivered: 2611
|
||||||
Full reservoir pulses remaining: 1023
|
Full reservoir pulses remaining: 1023
|
||||||
Time since activation: 3689
|
Time since activation: 3689
|
||||||
Alert 1 is InActive
|
Alert 1 is InActive
|
||||||
Alert 2 is InActive
|
Alert 2 is InActive
|
||||||
Alert 3 is InActive
|
Alert 3 is InActive
|
||||||
Alert 4 is InActive
|
Alert 4 is InActive
|
||||||
Alert 5 is InActive
|
Alert 5 is InActive
|
||||||
Alert 6 is InActive
|
Alert 6 is InActive
|
||||||
Alert 7 is InActive
|
Alert 7 is InActive
|
||||||
Occlusion alert active false
|
Occlusion alert active false
|
||||||
*/
|
*/
|
||||||
@Test @Throws(DecoderException::class) fun testValidResponseBolusPulsesRemaining() {
|
@Test @Throws(DecoderException::class) fun testValidResponseBolusPulsesRemaining() {
|
||||||
val encoded = Hex.decodeHex("1D180519C00E0039A7FF8085")
|
val encoded = Hex.decodeHex("1D180519C00E0039A7FF8085")
|
||||||
|
|
Loading…
Reference in a new issue