MockPumpManagerState.swift 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. //
  2. // MockPumpManagerState.swift
  3. // MockKit
  4. //
  5. // Created by Pete Schwamb on 7/31/19.
  6. // Copyright © 2019 LoopKit Authors. All rights reserved.
  7. //
  8. import Foundation
  9. import LoopKit
  10. public struct MockPumpManagerState: Equatable {
  11. public enum DeliverableIncrements: String, CaseIterable {
  12. case omnipod
  13. case medtronicX22
  14. case medtronicX23
  15. case custom
  16. var supportedBolusVolumes: [Double]? {
  17. switch self {
  18. case .omnipod:
  19. // 0.05 units for volumes between 0.05-30U
  20. return (1...600).map { Double($0) / Double(20) }
  21. case .medtronicX22:
  22. // 0.1 units for volumes between 0.1-25U
  23. return (1...250).map { Double($0) / Double(10) }
  24. case .medtronicX23:
  25. let breakpoints = [0, 1, 10, 25]
  26. let scales = [40, 20, 10]
  27. let scalingGroups = zip(scales, breakpoints.adjacentPairs().map(...))
  28. return scalingGroups.flatMap { (scale, range) -> [Double] in
  29. let scaledRanges = (range.lowerBound * scale + 1)...(range.upperBound * scale)
  30. return scaledRanges.map { Double($0) / Double(scale) }
  31. }
  32. case .custom:
  33. return nil
  34. }
  35. }
  36. var bolusVolumesDescription: String? {
  37. switch self {
  38. case .omnipod:
  39. // 0.05 units for volumes between 0.05-30U
  40. return "0.05-30 by 0.05"
  41. case .medtronicX22:
  42. // 0.1 units for volumes between 0.1-25U
  43. return "0.1-25 by 0.1"
  44. case .medtronicX23:
  45. // 0.025 units for rates between 0.0-0.975 U/h
  46. // 0.05 units for rates between 1-9.95 U/h
  47. // 0.1 units for rates between 10-25 U/h
  48. return "0-1-10-25 by 0.025|0.05|0.1"
  49. case .custom:
  50. return nil
  51. }
  52. }
  53. var supportedBasalRates: [Double]? {
  54. switch self {
  55. case .omnipod:
  56. // 0.05 units for rates between 0.05-30U/hr
  57. return (1...600).map { Double($0) / 20 }
  58. case .medtronicX22:
  59. // 0.05 units for rates between 0.0-35U/hr
  60. return (0...700).map { Double($0) / 20 }
  61. case .medtronicX23:
  62. // 0.025 units for rates between 0.0-0.975 U/h
  63. let rateGroup1 = (0...39).map { Double($0) / 40 }
  64. // 0.05 units for rates between 1-9.95 U/h
  65. let rateGroup2 = (20...199).map { Double($0) / 20 }
  66. // 0.1 units for rates between 10-35 U/h
  67. let rateGroup3 = (100...350).map { Double($0) / 10 }
  68. return rateGroup1 + rateGroup2 + rateGroup3
  69. case .custom:
  70. return nil
  71. }
  72. }
  73. public var basalRateDescription: String? {
  74. switch self {
  75. case .omnipod:
  76. // 0.05 units for rates between 0.05-30U/hr
  77. return "0.05-30 by 0.05"
  78. case .medtronicX22:
  79. // 0.05 units for rates between 0.0-35U/hr
  80. return "0-35 by 0.05"
  81. case .medtronicX23:
  82. // 0.025 units for rates between 0.0-0.975 U/h
  83. // 0.05 units for rates between 1-9.95 U/h
  84. // 0.1 units for rates between 10-35 U/h
  85. return "0-1-10-35 by 0.025|0.05|0.1"
  86. case .custom:
  87. return nil
  88. }
  89. }
  90. }
  91. public var deliverableIncrements: DeliverableIncrements {
  92. didSet {
  93. if let supportedBasalRates = deliverableIncrements.supportedBasalRates {
  94. self.supportedBasalRates = supportedBasalRates
  95. } else if let minBasalRates = supportedBasalRates.first, let maxBasalRates = supportedBasalRates.last, supportedBasalRates.indices.contains(1) {
  96. let stepSize = supportedBasalRates[1]-minBasalRates
  97. self.supportedBasalRates = (Int(minBasalRates/stepSize)...Int(maxBasalRates/stepSize)).map { Double($0) / (1 / stepSize) }
  98. }
  99. if let supportedBolusVolumes = deliverableIncrements.supportedBolusVolumes {
  100. self.supportedBolusVolumes = supportedBolusVolumes
  101. } else if let minBolusVolumes = supportedBolusVolumes.first, let maxBolusVolumes = supportedBolusVolumes.last, supportedBolusVolumes.indices.contains(1) {
  102. let stepSize = supportedBolusVolumes[1]-minBolusVolumes
  103. self.supportedBolusVolumes = (Int(minBolusVolumes/stepSize)...Int(maxBolusVolumes/stepSize)).map { Double($0) / (1 / stepSize) }
  104. }
  105. }
  106. }
  107. public var supportedBolusVolumes: [Double]
  108. public var supportedBolusVolumesDescription: String {
  109. guard let bolusVolumesDescription = deliverableIncrements.bolusVolumesDescription else {
  110. guard let minBolusVolume = supportedBolusVolumes.first, let maxBolusVolume = supportedBolusVolumes.last, supportedBolusVolumes.indices.contains(1) else {
  111. return "–"
  112. }
  113. return String(format: "\(minBolusVolume)-\(maxBolusVolume) by %.3f", (supportedBolusVolumes[1]-minBolusVolume))
  114. }
  115. return bolusVolumesDescription
  116. }
  117. public var supportedBasalRates: [Double]
  118. public var supportedBasalRatesDescription: String {
  119. guard let basalRatesDescription = deliverableIncrements.basalRateDescription else {
  120. guard let minBasalRate = supportedBasalRates.first, let maxBasalRate = supportedBasalRates.last, supportedBasalRates.indices.contains(1) else {
  121. return "–"
  122. }
  123. return String(format: "\(minBasalRate)-\(maxBasalRate) by %.3f", (supportedBasalRates[1]-minBasalRate))
  124. }
  125. return basalRatesDescription
  126. }
  127. public var basalRateSchedule: BasalRateSchedule?
  128. public var reservoirUnitsRemaining: Double
  129. public var tempBasalEnactmentShouldError: Bool
  130. public var bolusEnactmentShouldError: Bool
  131. public var bolusCancelShouldError: Bool
  132. public var deliverySuspensionShouldError: Bool
  133. public var deliveryResumptionShouldError: Bool
  134. public var bolusShouldCrash: Bool
  135. public var tempBasalShouldCrash: Bool
  136. public var deliveryCommandsShouldTriggerUncertainDelivery: Bool
  137. public var replacePumpComponent: Bool
  138. public var maximumBolus: Double
  139. public var maximumBasalRatePerHour: Double
  140. public var suspendState: SuspendState
  141. public var pumpBatteryChargeRemaining: Double?
  142. public var occlusionDetected: Bool = false
  143. public var pumpErrorDetected: Bool = false
  144. public var deliveryIsUncertain: Bool = false
  145. public var unfinalizedBolus: UnfinalizedDose?
  146. public var unfinalizedTempBasal: UnfinalizedDose?
  147. var finalizedDoses: [UnfinalizedDose]
  148. var additionalPumpEvents: [NewPumpEvent]
  149. public var progressPercentComplete: Double?
  150. public var progressWarningThresholdPercentValue: Double?
  151. public var progressCriticalThresholdPercentValue: Double?
  152. public var insulinType: InsulinType?
  153. public var timeZone: TimeZone
  154. public var dosesToStore: [UnfinalizedDose] {
  155. return finalizedDoses + [unfinalizedTempBasal, unfinalizedBolus].compactMap {$0}
  156. }
  157. public var pumpEventsToStore: [NewPumpEvent] {
  158. return dosesToStore.map { NewPumpEvent($0) } + additionalPumpEvents
  159. }
  160. public init(deliverableIncrements: DeliverableIncrements = .medtronicX22,
  161. reservoirUnitsRemaining: Double = 200.0,
  162. tempBasalEnactmentShouldError: Bool = false,
  163. bolusEnactmentShouldError: Bool = false,
  164. bolusCancelShouldError: Bool = false,
  165. deliverySuspensionShouldError: Bool = false,
  166. deliveryResumptionShouldError: Bool = false,
  167. bolusShouldCrash: Bool = false,
  168. tempBasalShouldCrash: Bool = false,
  169. deliveryCommandsShouldTriggerUncertainDelivery: Bool = false,
  170. replacePumpComponent: Bool = false,
  171. maximumBolus: Double = 25.0,
  172. maximumBasalRatePerHour: Double = 5.0,
  173. suspendState: SuspendState = .resumed(Date()),
  174. pumpBatteryChargeRemaining: Double? = 1,
  175. unfinalizedBolus: UnfinalizedDose? = nil,
  176. unfinalizedTempBasal: UnfinalizedDose? = nil,
  177. finalizedDoses: [UnfinalizedDose] = [],
  178. additionalPumpEvents: [NewPumpEvent] = [],
  179. progressWarningThresholdPercentValue: Double? = 0.75,
  180. progressCriticalThresholdPercentValue: Double? = 0.9,
  181. insulinType: InsulinType = .novolog)
  182. {
  183. self.deliverableIncrements = deliverableIncrements
  184. self.supportedBolusVolumes = deliverableIncrements.supportedBolusVolumes ?? []
  185. self.supportedBasalRates = deliverableIncrements.supportedBasalRates ?? []
  186. self.reservoirUnitsRemaining = reservoirUnitsRemaining
  187. self.tempBasalEnactmentShouldError = tempBasalEnactmentShouldError
  188. self.bolusEnactmentShouldError = bolusEnactmentShouldError
  189. self.bolusCancelShouldError = bolusCancelShouldError
  190. self.deliverySuspensionShouldError = deliverySuspensionShouldError
  191. self.deliveryResumptionShouldError = deliveryResumptionShouldError
  192. self.bolusShouldCrash = bolusShouldCrash
  193. self.tempBasalShouldCrash = tempBasalShouldCrash
  194. self.deliveryCommandsShouldTriggerUncertainDelivery = deliveryCommandsShouldTriggerUncertainDelivery
  195. self.replacePumpComponent = replacePumpComponent
  196. self.maximumBolus = maximumBolus
  197. self.maximumBasalRatePerHour = maximumBasalRatePerHour
  198. self.suspendState = suspendState
  199. self.pumpBatteryChargeRemaining = pumpBatteryChargeRemaining
  200. self.finalizedDoses = finalizedDoses
  201. self.additionalPumpEvents = additionalPumpEvents
  202. self.progressWarningThresholdPercentValue = progressWarningThresholdPercentValue
  203. self.progressCriticalThresholdPercentValue = progressCriticalThresholdPercentValue
  204. self.insulinType = insulinType
  205. self.timeZone = .currentFixed
  206. }
  207. public mutating func finalizeFinishedDoses() {
  208. if let bolus = unfinalizedBolus, bolus.finished {
  209. finalizedDoses.append(bolus)
  210. unfinalizedBolus = nil
  211. }
  212. if let tempBasal = unfinalizedTempBasal, tempBasal.finished {
  213. finalizedDoses.append(tempBasal)
  214. unfinalizedTempBasal = nil
  215. }
  216. }
  217. }
  218. extension MockPumpManagerState: RawRepresentable {
  219. public typealias RawValue = [String: Any]
  220. public init?(rawValue: RawValue) {
  221. guard let reservoirUnitsRemaining = rawValue["reservoirUnitsRemaining"] as? Double else {
  222. return nil
  223. }
  224. let defaultDeliverableIncrements: DeliverableIncrements = .medtronicX22
  225. self.deliverableIncrements = (rawValue["deliverableIncrements"] as? DeliverableIncrements.RawValue).flatMap(DeliverableIncrements.init(rawValue:)) ?? defaultDeliverableIncrements
  226. self.supportedBolusVolumes = rawValue["supportedBolusVolumes"] as? [Double] ?? defaultDeliverableIncrements.supportedBolusVolumes ?? []
  227. self.supportedBasalRates = rawValue["supportedBasalRates"] as? [Double] ?? defaultDeliverableIncrements.supportedBasalRates ?? []
  228. self.reservoirUnitsRemaining = reservoirUnitsRemaining
  229. self.tempBasalEnactmentShouldError = rawValue["tempBasalEnactmentShouldError"] as? Bool ?? false
  230. self.bolusEnactmentShouldError = rawValue["bolusEnactmentShouldError"] as? Bool ?? false
  231. self.bolusCancelShouldError = rawValue["bolusCancelShouldError"] as? Bool ?? false
  232. self.deliverySuspensionShouldError = rawValue["deliverySuspensionShouldError"] as? Bool ?? false
  233. self.deliveryResumptionShouldError = rawValue["deliveryResumptionShouldError"] as? Bool ?? false
  234. self.bolusShouldCrash = rawValue["bolusShouldCrash"] as? Bool ?? false
  235. self.tempBasalShouldCrash = rawValue["tempBasalShouldCrash"] as? Bool ?? false
  236. self.deliveryCommandsShouldTriggerUncertainDelivery = rawValue["deliveryCommandsShouldTriggerUncertainDelivery"] as? Bool ?? false
  237. self.replacePumpComponent = rawValue["replacePumpComponent"] as? Bool ?? false
  238. self.maximumBolus = rawValue["maximumBolus"] as? Double ?? 25.0
  239. self.maximumBasalRatePerHour = rawValue["maximumBasalRatePerHour"] as? Double ?? 5.0
  240. self.pumpBatteryChargeRemaining = rawValue["pumpBatteryChargeRemaining"] as? Double ?? nil
  241. self.occlusionDetected = rawValue["occlusionDetected"] as? Bool ?? false
  242. self.pumpErrorDetected = rawValue["pumpErrorDetected"] as? Bool ?? false
  243. self.deliveryIsUncertain = rawValue["deliveryIsUncertain"] as? Bool ?? false
  244. self.progressPercentComplete = rawValue["progressPercentComplete"] as? Double
  245. self.progressWarningThresholdPercentValue = rawValue["progressWarningThresholdPercentValue"] as? Double
  246. self.progressCriticalThresholdPercentValue = rawValue["progressCriticalThresholdPercentValue"] as? Double
  247. if let rawBasalRateSchedule = rawValue["basalRateSchedule"] as? BasalRateSchedule.RawValue {
  248. self.basalRateSchedule = BasalRateSchedule(rawValue: rawBasalRateSchedule)
  249. }
  250. if let rawUnfinalizedBolus = rawValue["unfinalizedBolus"] as? UnfinalizedDose.RawValue {
  251. self.unfinalizedBolus = UnfinalizedDose(rawValue: rawUnfinalizedBolus)
  252. }
  253. if let rawUnfinalizedTempBasal = rawValue["unfinalizedTempBasal"] as? UnfinalizedDose.RawValue {
  254. self.unfinalizedTempBasal = UnfinalizedDose(rawValue: rawUnfinalizedTempBasal)
  255. }
  256. if let rawFinalizedDoses = rawValue["finalizedDoses"] as? [UnfinalizedDose.RawValue] {
  257. self.finalizedDoses = rawFinalizedDoses.compactMap( { UnfinalizedDose(rawValue: $0) } )
  258. } else {
  259. self.finalizedDoses = []
  260. }
  261. if let rawAdditionalPumpEvents = rawValue["additionalPumpEvents"] as? [NewPumpEvent.RawValue] {
  262. self.additionalPumpEvents = rawAdditionalPumpEvents.compactMap( { NewPumpEvent(rawValue: $0) } )
  263. } else {
  264. self.additionalPumpEvents = []
  265. }
  266. if let rawSuspendState = rawValue["suspendState"] as? SuspendState.RawValue, let suspendState = SuspendState(rawValue: rawSuspendState) {
  267. self.suspendState = suspendState
  268. } else {
  269. self.suspendState = .resumed(Date())
  270. }
  271. if let rawInsulinType = rawValue["insulinType"] as? InsulinType.RawValue, let insulinType = InsulinType(rawValue: rawInsulinType) {
  272. self.insulinType = insulinType
  273. } else {
  274. self.insulinType = .novolog
  275. }
  276. if let timeZoneOffset = rawValue["timeZone"] as? Int {
  277. self.timeZone = TimeZone(secondsFromGMT: timeZoneOffset) ?? .currentFixed
  278. } else {
  279. self.timeZone = .currentFixed
  280. }
  281. }
  282. public var rawValue: RawValue {
  283. var raw: RawValue = [
  284. "deliverableIncrements": deliverableIncrements.rawValue,
  285. "supportedBolusVolumes": supportedBolusVolumes,
  286. "supportedBasalRates": supportedBasalRates,
  287. "reservoirUnitsRemaining": reservoirUnitsRemaining,
  288. "bolusShouldCrash": bolusShouldCrash,
  289. "tempBasalShouldCrash": tempBasalShouldCrash,
  290. "timeZone": timeZone.secondsFromGMT()
  291. ]
  292. raw["basalRateSchedule"] = basalRateSchedule?.rawValue
  293. raw["suspendState"] = suspendState.rawValue
  294. if tempBasalEnactmentShouldError {
  295. raw["tempBasalEnactmentShouldError"] = true
  296. }
  297. if bolusEnactmentShouldError {
  298. raw["bolusEnactmentShouldError"] = true
  299. }
  300. if bolusCancelShouldError {
  301. raw["bolusCancelShouldError"] = true
  302. }
  303. if deliverySuspensionShouldError {
  304. raw["deliverySuspensionShouldError"] = true
  305. }
  306. if deliveryResumptionShouldError {
  307. raw["deliveryResumptionShouldError"] = true
  308. }
  309. if deliveryCommandsShouldTriggerUncertainDelivery {
  310. raw["deliveryCommandsShouldTriggerUncertainDelivery"] = true
  311. }
  312. if replacePumpComponent {
  313. raw["replacePumpComponent"] = true
  314. }
  315. if deliveryIsUncertain {
  316. raw["deliveryIsUncertain"] = true
  317. }
  318. raw["finalizedDoses"] = finalizedDoses.map({ $0.rawValue })
  319. raw["additionalPumpEvents"] = additionalPumpEvents.map({ $0.rawValue })
  320. raw["maximumBolus"] = maximumBolus
  321. raw["maximumBasalRatePerHour"] = maximumBasalRatePerHour
  322. raw["unfinalizedBolus"] = unfinalizedBolus?.rawValue
  323. raw["unfinalizedTempBasal"] = unfinalizedTempBasal?.rawValue
  324. raw["pumpBatteryChargeRemaining"] = pumpBatteryChargeRemaining
  325. raw["occlusionDetected"] = occlusionDetected
  326. raw["pumpErrorDetected"] = pumpErrorDetected
  327. raw["progressPercentComplete"] = progressPercentComplete
  328. raw["progressWarningThresholdPercentValue"] = progressWarningThresholdPercentValue
  329. raw["progressCriticalThresholdPercentValue"] = progressCriticalThresholdPercentValue
  330. raw["insulinType"] = insulinType?.rawValue
  331. return raw
  332. }
  333. }
  334. extension MockPumpManagerState: CustomDebugStringConvertible {
  335. public var debugDescription: String {
  336. return """
  337. ## MockPumpManagerState
  338. * deliverableIncrements: \(deliverableIncrements)
  339. * reservoirUnitsRemaining: \(reservoirUnitsRemaining)
  340. * basalRateSchedule: \(basalRateSchedule as Any)
  341. * tempBasalEnactmentShouldError: \(tempBasalEnactmentShouldError)
  342. * bolusEnactmentShouldError: \(bolusEnactmentShouldError)
  343. * bolusCancelShouldError: \(bolusCancelShouldError)
  344. * deliverySuspensionShouldError: \(deliverySuspensionShouldError)
  345. * deliveryResumptionShouldError: \(deliveryResumptionShouldError)
  346. * maximumBolus: \(maximumBolus)
  347. * maximumBasalRatePerHour: \(maximumBasalRatePerHour)
  348. * pumpBatteryChargeRemaining: \(String(describing: pumpBatteryChargeRemaining))
  349. * suspendState: \(suspendState)
  350. * unfinalizedBolus: \(String(describing: unfinalizedBolus))
  351. * unfinalizedTempBasal: \(String(describing: unfinalizedTempBasal))
  352. * finalizedDoses: \(finalizedDoses)
  353. * additionalPumpEvents: \(additionalPumpEvents)
  354. * occlusionDetected: \(occlusionDetected)
  355. * pumpErrorDetected: \(pumpErrorDetected)
  356. * progressPercentComplete: \(progressPercentComplete as Any)
  357. * progressWarningThresholdPercentValue: \(progressWarningThresholdPercentValue as Any)
  358. * progressCriticalThresholdPercentValue: \(progressCriticalThresholdPercentValue as Any)
  359. * insulinType: \(insulinType as Any)
  360. """
  361. }
  362. }
  363. public enum SuspendState: Equatable, RawRepresentable {
  364. public typealias RawValue = [String: Any]
  365. private enum SuspendStateType: Int {
  366. case suspend, resume
  367. }
  368. case suspended(Date)
  369. case resumed(Date)
  370. public init?(rawValue: RawValue) {
  371. guard let suspendStateType = rawValue["suspendStateType"] as? SuspendStateType.RawValue,
  372. let date = rawValue["date"] as? Date else {
  373. return nil
  374. }
  375. switch SuspendStateType(rawValue: suspendStateType) {
  376. case .suspend?:
  377. self = .suspended(date)
  378. case .resume?:
  379. self = .resumed(date)
  380. default:
  381. return nil
  382. }
  383. }
  384. public var rawValue: RawValue {
  385. switch self {
  386. case .suspended(let date):
  387. return [
  388. "suspendStateType": SuspendStateType.suspend.rawValue,
  389. "date": date
  390. ]
  391. case .resumed(let date):
  392. return [
  393. "suspendStateType": SuspendStateType.resume.rawValue,
  394. "date": date
  395. ]
  396. }
  397. }
  398. }