DeviceDataManager.swift 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. import Combine
  2. import Foundation
  3. import LoopKit
  4. import LoopKitUI
  5. import MinimedKit
  6. import MockKit
  7. import OmniKit
  8. import SwiftDate
  9. import Swinject
  10. import UserNotifications
  11. protocol DeviceDataManager {
  12. var pumpManager: PumpManagerUI? { get set }
  13. var pumpDisplayState: CurrentValueSubject<PumpDisplayState?, Never> { get }
  14. var recommendsLoop: PassthroughSubject<Void, Never> { get }
  15. }
  16. private let staticPumpManagers: [PumpManagerUI.Type] = [
  17. MinimedPumpManager.self,
  18. OmnipodPumpManager.self,
  19. MockPumpManager.self
  20. ]
  21. private let staticPumpManagersByIdentifier: [String: PumpManagerUI.Type] = staticPumpManagers.reduce(into: [:]) { map, Type in
  22. map[Type.managerIdentifier] = Type
  23. }
  24. final class BaseDeviceDataManager: DeviceDataManager, Injectable {
  25. @Injected() private var pumpHistoryStorage: PumpHistoryStorage!
  26. @Injected() private var storage: FileStorage!
  27. @Persisted(key: "BaseDeviceDataManager.lastEventDate") var lastEventDate: Date? = nil
  28. let recommendsLoop = PassthroughSubject<Void, Never>()
  29. var pumpManager: PumpManagerUI? {
  30. didSet {
  31. pumpManager?.pumpManagerDelegate = self
  32. UserDefaults.standard.pumpManagerRawValue = pumpManager?.rawValue
  33. if let pumpManager = pumpManager {
  34. pumpDisplayState.value = PumpDisplayState(name: pumpManager.localizedTitle, image: pumpManager.smallImage)
  35. } else {
  36. pumpDisplayState.value = nil
  37. }
  38. }
  39. }
  40. let pumpDisplayState = CurrentValueSubject<PumpDisplayState?, Never>(nil)
  41. init(resolver: Resolver) {
  42. injectServices(resolver)
  43. setupPumpManager()
  44. }
  45. func setupPumpManager() {
  46. if let pumpManagerRawValue = UserDefaults.standard.pumpManagerRawValue {
  47. pumpManager = pumpManagerFromRawValue(pumpManagerRawValue)
  48. }
  49. }
  50. private func pumpManagerFromRawValue(_ rawValue: [String: Any]) -> PumpManagerUI? {
  51. guard let rawState = rawValue["state"] as? PumpManager.RawStateValue,
  52. let Manager = pumpManagerTypeFromRawValue(rawValue)
  53. else {
  54. return nil
  55. }
  56. return Manager.init(rawState: rawState) as? PumpManagerUI
  57. }
  58. private func pumpManagerTypeFromRawValue(_ rawValue: [String: Any]) -> PumpManager.Type? {
  59. guard let managerIdentifier = rawValue["managerIdentifier"] as? String else {
  60. return nil
  61. }
  62. return staticPumpManagersByIdentifier[managerIdentifier]
  63. }
  64. }
  65. extension BaseDeviceDataManager: PumpManagerDelegate {
  66. func pumpManager(_: PumpManager, didAdjustPumpClockBy _: TimeInterval) {
  67. // log.debug("didAdjustPumpClockBy %@", adjustment)
  68. }
  69. func pumpManagerDidUpdateState(_ pumpManager: PumpManager) {
  70. UserDefaults.standard.pumpManagerRawValue = pumpManager.rawValue
  71. }
  72. func pumpManagerBLEHeartbeatDidFire(_ pumpManager: PumpManager) {
  73. debug(.deviceManager, "Pump Heartbeat")
  74. pumpManager.ensureCurrentPumpData {
  75. debug(.deviceManager, "Pump Data updated")
  76. }
  77. }
  78. func pumpManagerMustProvideBLEHeartbeat(_: PumpManager) -> Bool {
  79. true
  80. }
  81. func pumpManager(_: PumpManager, didUpdate status: PumpManagerStatus, oldStatus _: PumpManagerStatus) {
  82. debug(.deviceManager, "New pump status Bolus: \(status.bolusState)")
  83. debug(.deviceManager, "New pump status Basal: \(String(describing: status.basalDeliveryState))")
  84. }
  85. func pumpManagerWillDeactivate(_: PumpManager) {
  86. pumpManager = nil
  87. }
  88. func pumpManager(_: PumpManager, didUpdatePumpRecordsBasalProfileStartEvents _: Bool) {}
  89. func pumpManager(_: PumpManager, didError error: PumpManagerError) {
  90. info(.deviceManager, "error: \(error.localizedDescription)")
  91. }
  92. func pumpManager(
  93. _: PumpManager,
  94. hasNewPumpEvents events: [NewPumpEvent],
  95. lastReconciliation _: Date?,
  96. completion: @escaping (_ error: Error?) -> Void
  97. ) {
  98. pumpHistoryStorage.storePumpEvents(events)
  99. lastEventDate = events.last?.date
  100. completion(nil)
  101. }
  102. func pumpManager(
  103. _: PumpManager,
  104. didReadReservoirValue units: Double,
  105. at date: Date,
  106. completion: @escaping (Result<
  107. (newValue: ReservoirValue, lastValue: ReservoirValue?, areStoredValuesContinuous: Bool),
  108. Error
  109. >) -> Void
  110. ) {
  111. debug(.deviceManager, "Reservoir Value \(units), at: \(date)")
  112. try? storage.save(Decimal(units), as: OpenAPS.Monitor.reservoir)
  113. let batteryPercent = Int((pumpManager?.status.pumpBatteryChargeRemaining ?? 1) * 100)
  114. let battery = Battery(percent: batteryPercent, voltage: nil, string: batteryPercent >= 10 ? .normal : .low)
  115. try? storage.save(battery, as: OpenAPS.Monitor.battery)
  116. completion(.success((
  117. newValue: Reservoir(startDate: Date(), unitVolume: units),
  118. lastValue: nil,
  119. areStoredValuesContinuous: true
  120. )))
  121. }
  122. func pumpManagerRecommendsLoop(_: PumpManager) {
  123. debug(.deviceManager, "Recomends loop")
  124. recommendsLoop.send()
  125. }
  126. func startDateToFilterNewPumpEvents(for _: PumpManager) -> Date {
  127. lastEventDate ?? Date().addingTimeInterval(-2.hours.timeInterval)
  128. }
  129. }
  130. // MARK: - DeviceManagerDelegate
  131. extension BaseDeviceDataManager: DeviceManagerDelegate {
  132. func scheduleNotification(
  133. for _: DeviceManager,
  134. identifier: String,
  135. content: UNNotificationContent,
  136. trigger: UNNotificationTrigger?
  137. ) {
  138. let request = UNNotificationRequest(
  139. identifier: identifier,
  140. content: content,
  141. trigger: trigger
  142. )
  143. DispatchQueue.main.async {
  144. UNUserNotificationCenter.current().add(request)
  145. }
  146. }
  147. func clearNotification(for _: DeviceManager, identifier: String) {
  148. DispatchQueue.main.async {
  149. UNUserNotificationCenter.current().removeDeliveredNotifications(withIdentifiers: [identifier])
  150. }
  151. }
  152. func removeNotificationRequests(for _: DeviceManager, identifiers: [String]) {
  153. DispatchQueue.main.async {
  154. UNUserNotificationCenter.current().removePendingNotificationRequests(withIdentifiers: identifiers)
  155. }
  156. }
  157. func deviceManager(
  158. _: DeviceManager,
  159. logEventForDeviceIdentifier _: String?,
  160. type _: DeviceLogEntryType,
  161. message _: String,
  162. completion _: ((Error?) -> Void)?
  163. ) {}
  164. }
  165. // MARK: - AlertPresenter
  166. extension BaseDeviceDataManager: AlertPresenter {
  167. func issueAlert(_: Alert) {}
  168. func retractAlert(identifier _: Alert.Identifier) {}
  169. }