BaseDeviceDataManager.swift 6.1 KB

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