MockPumpManagerState.swift 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  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 {
  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. public var progressPercentComplete: Double?
  146. public var progressWarningThresholdPercentValue: Double?
  147. public var progressCriticalThresholdPercentValue: Double?
  148. public var insulinType: InsulinType
  149. public var dosesToStore: [UnfinalizedDose] {
  150. return finalizedDoses + [unfinalizedTempBasal, unfinalizedBolus].compactMap {$0}
  151. }
  152. public mutating func finalizeFinishedDoses() {
  153. if let bolus = unfinalizedBolus, bolus.finished {
  154. finalizedDoses.append(bolus)
  155. unfinalizedBolus = nil
  156. }
  157. if let tempBasal = unfinalizedTempBasal, tempBasal.finished {
  158. finalizedDoses.append(tempBasal)
  159. unfinalizedTempBasal = nil
  160. }
  161. }
  162. }
  163. extension MockPumpManagerState: RawRepresentable {
  164. public typealias RawValue = [String: Any]
  165. public init?(rawValue: RawValue) {
  166. guard let reservoirUnitsRemaining = rawValue["reservoirUnitsRemaining"] as? Double else {
  167. return nil
  168. }
  169. let defaultDeliverableIncrements: DeliverableIncrements = .medtronicX22
  170. self.deliverableIncrements = (rawValue["deliverableIncrements"] as? DeliverableIncrements.RawValue).flatMap(DeliverableIncrements.init(rawValue:)) ?? defaultDeliverableIncrements
  171. self.supportedBolusVolumes = rawValue["supportedBolusVolumes"] as? [Double] ?? defaultDeliverableIncrements.supportedBolusVolumes ?? []
  172. self.supportedBasalRates = rawValue["supportedBasalRates"] as? [Double] ?? defaultDeliverableIncrements.supportedBasalRates ?? []
  173. self.reservoirUnitsRemaining = reservoirUnitsRemaining
  174. self.tempBasalEnactmentShouldError = rawValue["tempBasalEnactmentShouldError"] as? Bool ?? false
  175. self.bolusEnactmentShouldError = rawValue["bolusEnactmentShouldError"] as? Bool ?? false
  176. self.bolusCancelShouldError = rawValue["bolusCancelShouldError"] as? Bool ?? false
  177. self.deliverySuspensionShouldError = rawValue["deliverySuspensionShouldError"] as? Bool ?? false
  178. self.deliveryResumptionShouldError = rawValue["deliveryResumptionShouldError"] as? Bool ?? false
  179. self.deliveryCommandsShouldTriggerUncertainDelivery = rawValue["deliveryCommandsShouldTriggerUncertainDelivery"] as? Bool ?? false
  180. self.maximumBolus = rawValue["maximumBolus"] as? Double ?? 25.0
  181. self.maximumBasalRatePerHour = rawValue["maximumBasalRatePerHour"] as? Double ?? 5.0
  182. self.pumpBatteryChargeRemaining = rawValue["pumpBatteryChargeRemaining"] as? Double ?? nil
  183. self.occlusionDetected = rawValue["occlusionDetected"] as? Bool ?? false
  184. self.pumpErrorDetected = rawValue["pumpErrorDetected"] as? Bool ?? false
  185. self.deliveryIsUncertain = rawValue["deliveryIsUncertain"] as? Bool ?? false
  186. self.progressPercentComplete = rawValue["progressPercentComplete"] as? Double
  187. self.progressWarningThresholdPercentValue = rawValue["progressWarningThresholdPercentValue"] as? Double
  188. self.progressCriticalThresholdPercentValue = rawValue["progressCriticalThresholdPercentValue"] as? Double
  189. if let rawBasalRateSchedule = rawValue["basalRateSchedule"] as? BasalRateSchedule.RawValue {
  190. self.basalRateSchedule = BasalRateSchedule(rawValue: rawBasalRateSchedule)
  191. }
  192. if let rawUnfinalizedBolus = rawValue["unfinalizedBolus"] as? UnfinalizedDose.RawValue {
  193. self.unfinalizedBolus = UnfinalizedDose(rawValue: rawUnfinalizedBolus)
  194. }
  195. if let rawUnfinalizedTempBasal = rawValue["unfinalizedTempBasal"] as? UnfinalizedDose.RawValue {
  196. self.unfinalizedTempBasal = UnfinalizedDose(rawValue: rawUnfinalizedTempBasal)
  197. }
  198. if let rawFinalizedDoses = rawValue["finalizedDoses"] as? [UnfinalizedDose.RawValue] {
  199. self.finalizedDoses = rawFinalizedDoses.compactMap( { UnfinalizedDose(rawValue: $0) } )
  200. } else {
  201. self.finalizedDoses = []
  202. }
  203. if let rawSuspendState = rawValue["suspendState"] as? SuspendState.RawValue, let suspendState = SuspendState(rawValue: rawSuspendState) {
  204. self.suspendState = suspendState
  205. } else {
  206. self.suspendState = .resumed(Date())
  207. }
  208. if let rawInsulinType = rawValue["insulinType"] as? InsulinType.RawValue, let insulinType = InsulinType(rawValue: rawInsulinType) {
  209. self.insulinType = insulinType
  210. } else {
  211. self.insulinType = .novolog
  212. }
  213. }
  214. public var rawValue: RawValue {
  215. var raw: RawValue = [
  216. "deliverableIncrements": deliverableIncrements.rawValue,
  217. "supportedBolusVolumes": supportedBolusVolumes,
  218. "supportedBasalRates": supportedBasalRates,
  219. "reservoirUnitsRemaining": reservoirUnitsRemaining,
  220. "insulinType": insulinType.rawValue
  221. ]
  222. raw["basalRateSchedule"] = basalRateSchedule?.rawValue
  223. raw["suspendState"] = suspendState.rawValue
  224. if tempBasalEnactmentShouldError {
  225. raw["tempBasalEnactmentShouldError"] = true
  226. }
  227. if bolusEnactmentShouldError {
  228. raw["bolusEnactmentShouldError"] = true
  229. }
  230. if bolusCancelShouldError {
  231. raw["bolusCancelShouldError"] = true
  232. }
  233. if deliverySuspensionShouldError {
  234. raw["deliverySuspensionShouldError"] = true
  235. }
  236. if deliveryResumptionShouldError {
  237. raw["deliveryResumptionShouldError"] = true
  238. }
  239. if deliveryCommandsShouldTriggerUncertainDelivery {
  240. raw["deliveryCommandsShouldTriggerUncertainDelivery"] = true
  241. }
  242. if deliveryIsUncertain {
  243. raw["deliveryIsUncertain"] = true
  244. }
  245. raw["finalizedDoses"] = finalizedDoses.map( { $0.rawValue })
  246. raw["maximumBolus"] = maximumBolus
  247. raw["maximumBasalRatePerHour"] = maximumBasalRatePerHour
  248. raw["unfinalizedBolus"] = unfinalizedBolus?.rawValue
  249. raw["unfinalizedTempBasal"] = unfinalizedTempBasal?.rawValue
  250. raw["pumpBatteryChargeRemaining"] = pumpBatteryChargeRemaining
  251. raw["occlusionDetected"] = occlusionDetected
  252. raw["pumpErrorDetected"] = pumpErrorDetected
  253. raw["progressPercentComplete"] = progressPercentComplete
  254. raw["progressWarningThresholdPercentValue"] = progressWarningThresholdPercentValue
  255. raw["progressCriticalThresholdPercentValue"] = progressCriticalThresholdPercentValue
  256. return raw
  257. }
  258. }
  259. extension MockPumpManagerState: CustomDebugStringConvertible {
  260. public var debugDescription: String {
  261. return """
  262. ## MockPumpManagerState
  263. * deliverableIncrements: \(deliverableIncrements)
  264. * reservoirUnitsRemaining: \(reservoirUnitsRemaining)
  265. * basalRateSchedule: \(basalRateSchedule as Any)
  266. * tempBasalEnactmentShouldError: \(tempBasalEnactmentShouldError)
  267. * bolusEnactmentShouldError: \(bolusEnactmentShouldError)
  268. * bolusCancelShouldError: \(bolusCancelShouldError)
  269. * deliverySuspensionShouldError: \(deliverySuspensionShouldError)
  270. * deliveryResumptionShouldError: \(deliveryResumptionShouldError)
  271. * maximumBolus: \(maximumBolus)
  272. * maximumBasalRatePerHour: \(maximumBasalRatePerHour)
  273. * pumpBatteryChargeRemaining: \(String(describing: pumpBatteryChargeRemaining))
  274. * suspendState: \(suspendState)
  275. * unfinalizedBolus: \(String(describing: unfinalizedBolus))
  276. * unfinalizedTempBasal: \(String(describing: unfinalizedTempBasal))
  277. * finalizedDoses: \(finalizedDoses)
  278. * occlusionDetected: \(occlusionDetected)
  279. * pumpErrorDetected: \(pumpErrorDetected)
  280. * progressPercentComplete: \(progressPercentComplete as Any)
  281. * progressWarningThresholdPercentValue: \(progressWarningThresholdPercentValue as Any)
  282. * progressCriticalThresholdPercentValue: \(progressCriticalThresholdPercentValue as Any)
  283. """
  284. }
  285. }
  286. public enum SuspendState: Equatable, RawRepresentable {
  287. public typealias RawValue = [String: Any]
  288. private enum SuspendStateType: Int {
  289. case suspend, resume
  290. }
  291. case suspended(Date)
  292. case resumed(Date)
  293. public init?(rawValue: RawValue) {
  294. guard let suspendStateType = rawValue["suspendStateType"] as? SuspendStateType.RawValue,
  295. let date = rawValue["date"] as? Date else {
  296. return nil
  297. }
  298. switch SuspendStateType(rawValue: suspendStateType) {
  299. case .suspend?:
  300. self = .suspended(date)
  301. case .resume?:
  302. self = .resumed(date)
  303. default:
  304. return nil
  305. }
  306. }
  307. public var rawValue: RawValue {
  308. switch self {
  309. case .suspended(let date):
  310. return [
  311. "suspendStateType": SuspendStateType.suspend.rawValue,
  312. "date": date
  313. ]
  314. case .resumed(let date):
  315. return [
  316. "suspendStateType": SuspendStateType.resume.rawValue,
  317. "date": date
  318. ]
  319. }
  320. }
  321. }