DeviceDataManager.swift 7.3 KB

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