MockPumpManagerState.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  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) * 0.05 }
  21. case .medtronicX22:
  22. // 0.1 units for volumes between 0.1-25U
  23. return (1...250).map { Double($0) * 0.1 }
  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 deliveryCommandsShouldTriggerUncertainDelivery: Bool
  135. public var maximumBolus: Double
  136. public var maximumBasalRatePerHour: Double
  137. public var suspendState: SuspendState
  138. public var pumpBatteryChargeRemaining: Double?
  139. public var occlusionDetected: Bool = false
  140. public var pumpErrorDetected: Bool = false
  141. public var deliveryIsUncertain: Bool = false
  142. public var unfinalizedBolus: UnfinalizedDose?
  143. public var unfinalizedTempBasal: UnfinalizedDose?
  144. var finalizedDoses: [UnfinalizedDose]
  145. var additionalPumpEvents: [NewPumpEvent]
  146. public var progressPercentComplete: Double?
  147. public var progressWarningThresholdPercentValue: Double?
  148. public var progressCriticalThresholdPercentValue: Double?
  149. public var insulinType: InsulinType?
  150. public var timeZone: TimeZone
  151. public var dosesToStore: [UnfinalizedDose] {
  152. return finalizedDoses + [unfinalizedTempBasal, unfinalizedBolus].compactMap {$0}
  153. }
  154. public var pumpEventsToStore: [NewPumpEvent] {
  155. return dosesToStore.map { NewPumpEvent($0) } + additionalPumpEvents
  156. }
  157. public init(deliverableIncrements: DeliverableIncrements = .medtronicX22,
  158. reservoirUnitsRemaining: Double = 200.0,
  159. tempBasalEnactmentShouldError: Bool = false,
  160. bolusEnactmentShouldError: Bool = false,
  161. bolusCancelShouldError: Bool = false,
  162. deliverySuspensionShouldError: Bool = false,
  163. deliveryResumptionShouldError: Bool = false,
  164. deliveryCommandsShouldTriggerUncertainDelivery: Bool = false,
  165. maximumBolus: Double = 25.0,
  166. maximumBasalRatePerHour: Double = 5.0,
  167. suspendState: SuspendState = .resumed(Date()),
  168. pumpBatteryChargeRemaining: Double? = 1,
  169. unfinalizedBolus: UnfinalizedDose? = nil,
  170. unfinalizedTempBasal: UnfinalizedDose? = nil,
  171. finalizedDoses: [UnfinalizedDose] = [],
  172. additionalPumpEvents: [NewPumpEvent] = [],
  173. progressWarningThresholdPercentValue: Double? = 0.75,
  174. progressCriticalThresholdPercentValue: Double? = 0.9,
  175. insulinType: InsulinType = .novolog)
  176. {
  177. self.deliverableIncrements = deliverableIncrements
  178. self.supportedBolusVolumes = deliverableIncrements.supportedBolusVolumes ?? []
  179. self.supportedBasalRates = deliverableIncrements.supportedBasalRates ?? []
  180. self.reservoirUnitsRemaining = reservoirUnitsRemaining
  181. self.tempBasalEnactmentShouldError = tempBasalEnactmentShouldError
  182. self.bolusEnactmentShouldError = bolusEnactmentShouldError
  183. self.bolusCancelShouldError = bolusCancelShouldError
  184. self.deliverySuspensionShouldError = deliverySuspensionShouldError
  185. self.deliveryResumptionShouldError = deliveryResumptionShouldError
  186. self.deliveryCommandsShouldTriggerUncertainDelivery = deliveryCommandsShouldTriggerUncertainDelivery
  187. self.maximumBolus = maximumBolus
  188. self.maximumBasalRatePerHour = maximumBasalRatePerHour
  189. self.suspendState = suspendState
  190. self.pumpBatteryChargeRemaining = pumpBatteryChargeRemaining
  191. self.finalizedDoses = finalizedDoses
  192. self.additionalPumpEvents = additionalPumpEvents
  193. self.progressWarningThresholdPercentValue = progressWarningThresholdPercentValue
  194. self.progressCriticalThresholdPercentValue = progressCriticalThresholdPercentValue
  195. self.insulinType = insulinType
  196. self.timeZone = .currentFixed
  197. }
  198. public mutating func finalizeFinishedDoses() {
  199. if let bolus = unfinalizedBolus, bolus.finished {
  200. finalizedDoses.append(bolus)
  201. unfinalizedBolus = nil
  202. }
  203. if let tempBasal = unfinalizedTempBasal, tempBasal.finished {
  204. finalizedDoses.append(tempBasal)
  205. unfinalizedTempBasal = nil
  206. }
  207. }
  208. }
  209. extension MockPumpManagerState: RawRepresentable {
  210. public typealias RawValue = [String: Any]
  211. public init?(rawValue: RawValue) {
  212. guard let reservoirUnitsRemaining = rawValue["reservoirUnitsRemaining"] as? Double else {
  213. return nil
  214. }
  215. let defaultDeliverableIncrements: DeliverableIncrements = .medtronicX22
  216. self.deliverableIncrements = (rawValue["deliverableIncrements"] as? DeliverableIncrements.RawValue).flatMap(DeliverableIncrements.init(rawValue:)) ?? defaultDeliverableIncrements
  217. self.supportedBolusVolumes = rawValue["supportedBolusVolumes"] as? [Double] ?? defaultDeliverableIncrements.supportedBolusVolumes ?? []
  218. self.supportedBasalRates = rawValue["supportedBasalRates"] as? [Double] ?? defaultDeliverableIncrements.supportedBasalRates ?? []
  219. self.reservoirUnitsRemaining = reservoirUnitsRemaining
  220. self.tempBasalEnactmentShouldError = rawValue["tempBasalEnactmentShouldError"] as? Bool ?? false
  221. self.bolusEnactmentShouldError = rawValue["bolusEnactmentShouldError"] as? Bool ?? false
  222. self.bolusCancelShouldError = rawValue["bolusCancelShouldError"] as? Bool ?? false
  223. self.deliverySuspensionShouldError = rawValue["deliverySuspensionShouldError"] as? Bool ?? false
  224. self.deliveryResumptionShouldError = rawValue["deliveryResumptionShouldError"] as? Bool ?? false
  225. self.deliveryCommandsShouldTriggerUncertainDelivery = rawValue["deliveryCommandsShouldTriggerUncertainDelivery"] as? Bool ?? false
  226. self.maximumBolus = rawValue["maximumBolus"] as? Double ?? 25.0
  227. self.maximumBasalRatePerHour = rawValue["maximumBasalRatePerHour"] as? Double ?? 5.0
  228. self.pumpBatteryChargeRemaining = rawValue["pumpBatteryChargeRemaining"] as? Double ?? nil
  229. self.occlusionDetected = rawValue["occlusionDetected"] as? Bool ?? false
  230. self.pumpErrorDetected = rawValue["pumpErrorDetected"] as? Bool ?? false
  231. self.deliveryIsUncertain = rawValue["deliveryIsUncertain"] as? Bool ?? false
  232. self.progressPercentComplete = rawValue["progressPercentComplete"] as? Double
  233. self.progressWarningThresholdPercentValue = rawValue["progressWarningThresholdPercentValue"] as? Double
  234. self.progressCriticalThresholdPercentValue = rawValue["progressCriticalThresholdPercentValue"] as? Double
  235. if let rawBasalRateSchedule = rawValue["basalRateSchedule"] as? BasalRateSchedule.RawValue {
  236. self.basalRateSchedule = BasalRateSchedule(rawValue: rawBasalRateSchedule)
  237. }
  238. if let rawUnfinalizedBolus = rawValue["unfinalizedBolus"] as? UnfinalizedDose.RawValue {
  239. self.unfinalizedBolus = UnfinalizedDose(rawValue: rawUnfinalizedBolus)
  240. }
  241. if let rawUnfinalizedTempBasal = rawValue["unfinalizedTempBasal"] as? UnfinalizedDose.RawValue {
  242. self.unfinalizedTempBasal = UnfinalizedDose(rawValue: rawUnfinalizedTempBasal)
  243. }
  244. if let rawFinalizedDoses = rawValue["finalizedDoses"] as? [UnfinalizedDose.RawValue] {
  245. self.finalizedDoses = rawFinalizedDoses.compactMap( { UnfinalizedDose(rawValue: $0) } )
  246. } else {
  247. self.finalizedDoses = []
  248. }
  249. if let rawAdditionalPumpEvents = rawValue["additionalPumpEvents"] as? [NewPumpEvent.RawValue] {
  250. self.additionalPumpEvents = rawAdditionalPumpEvents.compactMap( { NewPumpEvent(rawValue: $0) } )
  251. } else {
  252. self.additionalPumpEvents = []
  253. }
  254. if let rawSuspendState = rawValue["suspendState"] as? SuspendState.RawValue, let suspendState = SuspendState(rawValue: rawSuspendState) {
  255. self.suspendState = suspendState
  256. } else {
  257. self.suspendState = .resumed(Date())
  258. }
  259. if let rawInsulinType = rawValue["insulinType"] as? InsulinType.RawValue, let insulinType = InsulinType(rawValue: rawInsulinType) {
  260. self.insulinType = insulinType
  261. } else {
  262. self.insulinType = .novolog
  263. }
  264. if let timeZoneOffset = rawValue["timeZone"] as? Int {
  265. self.timeZone = TimeZone(secondsFromGMT: timeZoneOffset) ?? .currentFixed
  266. } else {
  267. self.timeZone = .currentFixed
  268. }
  269. }
  270. public var rawValue: RawValue {
  271. var raw: RawValue = [
  272. "deliverableIncrements": deliverableIncrements.rawValue,
  273. "supportedBolusVolumes": supportedBolusVolumes,
  274. "supportedBasalRates": supportedBasalRates,
  275. "reservoirUnitsRemaining": reservoirUnitsRemaining,
  276. "timeZone": timeZone.secondsFromGMT()
  277. ]
  278. raw["basalRateSchedule"] = basalRateSchedule?.rawValue
  279. raw["suspendState"] = suspendState.rawValue
  280. if tempBasalEnactmentShouldError {
  281. raw["tempBasalEnactmentShouldError"] = true
  282. }
  283. if bolusEnactmentShouldError {
  284. raw["bolusEnactmentShouldError"] = true
  285. }
  286. if bolusCancelShouldError {
  287. raw["bolusCancelShouldError"] = true
  288. }
  289. if deliverySuspensionShouldError {
  290. raw["deliverySuspensionShouldError"] = true
  291. }
  292. if deliveryResumptionShouldError {
  293. raw["deliveryResumptionShouldError"] = true
  294. }
  295. if deliveryCommandsShouldTriggerUncertainDelivery {
  296. raw["deliveryCommandsShouldTriggerUncertainDelivery"] = true
  297. }
  298. if deliveryIsUncertain {
  299. raw["deliveryIsUncertain"] = true
  300. }
  301. raw["finalizedDoses"] = finalizedDoses.map({ $0.rawValue })
  302. raw["additionalPumpEvents"] = additionalPumpEvents.map({ $0.rawValue })
  303. raw["maximumBolus"] = maximumBolus
  304. raw["maximumBasalRatePerHour"] = maximumBasalRatePerHour
  305. raw["unfinalizedBolus"] = unfinalizedBolus?.rawValue
  306. raw["unfinalizedTempBasal"] = unfinalizedTempBasal?.rawValue
  307. raw["pumpBatteryChargeRemaining"] = pumpBatteryChargeRemaining
  308. raw["occlusionDetected"] = occlusionDetected
  309. raw["pumpErrorDetected"] = pumpErrorDetected
  310. raw["progressPercentComplete"] = progressPercentComplete
  311. raw["progressWarningThresholdPercentValue"] = progressWarningThresholdPercentValue
  312. raw["progressCriticalThresholdPercentValue"] = progressCriticalThresholdPercentValue
  313. raw["insulinType"] = insulinType?.rawValue
  314. return raw
  315. }
  316. }
  317. extension MockPumpManagerState: CustomDebugStringConvertible {
  318. public var debugDescription: String {
  319. return """
  320. ## MockPumpManagerState
  321. * deliverableIncrements: \(deliverableIncrements)
  322. * reservoirUnitsRemaining: \(reservoirUnitsRemaining)
  323. * basalRateSchedule: \(basalRateSchedule as Any)
  324. * tempBasalEnactmentShouldError: \(tempBasalEnactmentShouldError)
  325. * bolusEnactmentShouldError: \(bolusEnactmentShouldError)
  326. * bolusCancelShouldError: \(bolusCancelShouldError)
  327. * deliverySuspensionShouldError: \(deliverySuspensionShouldError)
  328. * deliveryResumptionShouldError: \(deliveryResumptionShouldError)
  329. * maximumBolus: \(maximumBolus)
  330. * maximumBasalRatePerHour: \(maximumBasalRatePerHour)
  331. * pumpBatteryChargeRemaining: \(String(describing: pumpBatteryChargeRemaining))
  332. * suspendState: \(suspendState)
  333. * unfinalizedBolus: \(String(describing: unfinalizedBolus))
  334. * unfinalizedTempBasal: \(String(describing: unfinalizedTempBasal))
  335. * finalizedDoses: \(finalizedDoses)
  336. * additionalPumpEvents: \(additionalPumpEvents)
  337. * occlusionDetected: \(occlusionDetected)
  338. * pumpErrorDetected: \(pumpErrorDetected)
  339. * progressPercentComplete: \(progressPercentComplete as Any)
  340. * progressWarningThresholdPercentValue: \(progressWarningThresholdPercentValue as Any)
  341. * progressCriticalThresholdPercentValue: \(progressCriticalThresholdPercentValue as Any)
  342. * insulinType: \(insulinType as Any)
  343. """
  344. }
  345. }
  346. public enum SuspendState: Equatable, RawRepresentable {
  347. public typealias RawValue = [String: Any]
  348. private enum SuspendStateType: Int {
  349. case suspend, resume
  350. }
  351. case suspended(Date)
  352. case resumed(Date)
  353. public init?(rawValue: RawValue) {
  354. guard let suspendStateType = rawValue["suspendStateType"] as? SuspendStateType.RawValue,
  355. let date = rawValue["date"] as? Date else {
  356. return nil
  357. }
  358. switch SuspendStateType(rawValue: suspendStateType) {
  359. case .suspend?:
  360. self = .suspended(date)
  361. case .resume?:
  362. self = .resumed(date)
  363. default:
  364. return nil
  365. }
  366. }
  367. public var rawValue: RawValue {
  368. switch self {
  369. case .suspended(let date):
  370. return [
  371. "suspendStateType": SuspendStateType.suspend.rawValue,
  372. "date": date
  373. ]
  374. case .resumed(let date):
  375. return [
  376. "suspendStateType": SuspendStateType.resume.rawValue,
  377. "date": date
  378. ]
  379. }
  380. }
  381. }