FetchGlucoseManager.swift 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. import Combine
  2. import Foundation
  3. import SwiftDate
  4. import Swinject
  5. import UIKit
  6. protocol FetchGlucoseManager: SourceInfoProvider {
  7. func updateGlucoseStore(newBloodGlucose: [BloodGlucose])
  8. func refreshCGM()
  9. func updateGlucoseSource()
  10. var glucoseSource: GlucoseSource! { get }
  11. var cgmGlucoseSourceType: CGMType? { get set }
  12. var settingsManager: SettingsManager! { get }
  13. }
  14. final class BaseFetchGlucoseManager: FetchGlucoseManager, Injectable {
  15. private let processQueue = DispatchQueue(label: "BaseGlucoseManager.processQueue")
  16. @Injected() var glucoseStorage: GlucoseStorage!
  17. @Injected() var nightscoutManager: NightscoutManager!
  18. @Injected() var apsManager: APSManager!
  19. @Injected() var settingsManager: SettingsManager!
  20. @Injected() var libreTransmitter: LibreTransmitterSource!
  21. @Injected() var healthKitManager: HealthKitManager!
  22. @Injected() var deviceDataManager: DeviceDataManager!
  23. private var lifetime = Lifetime()
  24. private let timer = DispatchTimer(timeInterval: 1.minutes.timeInterval)
  25. var cgmGlucoseSourceType: CGMType?
  26. private lazy var dexcomSourceG5 = DexcomSourceG5(glucoseStorage: glucoseStorage, glucoseManager: self)
  27. private lazy var dexcomSourceG6 = DexcomSourceG6(glucoseStorage: glucoseStorage, glucoseManager: self)
  28. private lazy var dexcomSourceG7 = DexcomSourceG7(glucoseStorage: glucoseStorage, glucoseManager: self)
  29. private lazy var simulatorSource = GlucoseSimulatorSource()
  30. init(resolver: Resolver) {
  31. injectServices(resolver)
  32. updateGlucoseSource()
  33. subscribe()
  34. }
  35. var glucoseSource: GlucoseSource!
  36. func updateGlucoseSource() {
  37. switch settingsManager.settings.cgm {
  38. case .xdrip:
  39. glucoseSource = AppGroupSource(from: "xDrip", cgmType: .xdrip)
  40. case .dexcomG5:
  41. glucoseSource = dexcomSourceG5
  42. case .dexcomG6:
  43. glucoseSource = dexcomSourceG6
  44. case .dexcomG7:
  45. glucoseSource = dexcomSourceG7
  46. case .nightscout:
  47. glucoseSource = nightscoutManager
  48. case .simulator:
  49. glucoseSource = simulatorSource
  50. case .libreTransmitter:
  51. glucoseSource = libreTransmitter
  52. case .glucoseDirect:
  53. glucoseSource = AppGroupSource(from: "GlucoseDirect", cgmType: .glucoseDirect)
  54. case .enlite:
  55. glucoseSource = deviceDataManager
  56. }
  57. // update the config
  58. cgmGlucoseSourceType = settingsManager.settings.cgm
  59. if settingsManager.settings.cgm != .libreTransmitter {
  60. libreTransmitter.manager = nil
  61. } else {
  62. libreTransmitter.glucoseManager = self
  63. }
  64. }
  65. /// function called when a callback is fired by CGM BLE - no more used
  66. public func updateGlucoseStore(newBloodGlucose: [BloodGlucose]) {
  67. let syncDate = glucoseStorage.syncDate()
  68. debug(.deviceManager, "CGM BLE FETCHGLUCOSE : SyncDate is \(syncDate)")
  69. glucoseStoreAndHeartDecision(syncDate: syncDate, glucose: newBloodGlucose)
  70. }
  71. /// function to try to force the refresh of the CGM - generally provide by the pump heartbeat
  72. public func refreshCGM() {
  73. debug(.deviceManager, "refreshCGM by pump")
  74. updateGlucoseSource()
  75. Publishers.CombineLatest3(
  76. Just(glucoseStorage.syncDate()),
  77. healthKitManager.fetch(nil),
  78. glucoseSource.fetchIfNeeded()
  79. )
  80. .eraseToAnyPublisher()
  81. .receive(on: processQueue)
  82. .sink { syncDate, glucoseFromHealth, glucose in
  83. debug(.nightscout, "refreshCGM FETCHGLUCOSE : SyncDate is \(syncDate)")
  84. self.glucoseStoreAndHeartDecision(syncDate: syncDate, glucose: glucose, glucoseFromHealth: glucoseFromHealth)
  85. }
  86. .store(in: &lifetime)
  87. }
  88. private func glucoseStoreAndHeartDecision(syncDate: Date, glucose: [BloodGlucose], glucoseFromHealth: [BloodGlucose] = []) {
  89. let allGlucose = glucose + glucoseFromHealth
  90. var filteredByDate: [BloodGlucose] = []
  91. var filtered: [BloodGlucose] = []
  92. // start background time extension
  93. var backGroundFetchBGTaskID: UIBackgroundTaskIdentifier?
  94. backGroundFetchBGTaskID = UIApplication.shared.beginBackgroundTask(withName: "save BG starting") {
  95. guard let bg = backGroundFetchBGTaskID else { return }
  96. UIApplication.shared.endBackgroundTask(bg)
  97. backGroundFetchBGTaskID = .invalid
  98. }
  99. guard allGlucose.isNotEmpty else {
  100. if let backgroundTask = backGroundFetchBGTaskID {
  101. UIApplication.shared.endBackgroundTask(backgroundTask)
  102. backGroundFetchBGTaskID = .invalid
  103. }
  104. return
  105. }
  106. filteredByDate = allGlucose.filter { $0.dateString > syncDate }
  107. filtered = glucoseStorage.filterTooFrequentGlucose(filteredByDate, at: syncDate)
  108. guard filtered.isNotEmpty else {
  109. // end of the BG tasks
  110. if let backgroundTask = backGroundFetchBGTaskID {
  111. UIApplication.shared.endBackgroundTask(backgroundTask)
  112. backGroundFetchBGTaskID = .invalid
  113. }
  114. return
  115. }
  116. debug(.deviceManager, "New glucose found")
  117. glucoseStorage.storeGlucose(filtered)
  118. deviceDataManager.heartbeat(date: Date())
  119. nightscoutManager.uploadGlucose()
  120. let glucoseForHealth = filteredByDate.filter { !glucoseFromHealth.contains($0) }
  121. guard glucoseForHealth.isNotEmpty else {
  122. // end of the BG tasks
  123. if let backgroundTask = backGroundFetchBGTaskID {
  124. UIApplication.shared.endBackgroundTask(backgroundTask)
  125. backGroundFetchBGTaskID = .invalid
  126. }
  127. return
  128. }
  129. healthKitManager.saveIfNeeded(bloodGlucose: glucoseForHealth)
  130. // end of the BG tasks
  131. if let backgroundTask = backGroundFetchBGTaskID {
  132. UIApplication.shared.endBackgroundTask(backgroundTask)
  133. backGroundFetchBGTaskID = .invalid
  134. }
  135. }
  136. /// The function used to start the timer sync - Function of the variable defined in config
  137. private func subscribe() {
  138. timer.publisher
  139. .receive(on: processQueue)
  140. .flatMap { _ -> AnyPublisher<[BloodGlucose], Never> in
  141. debug(.nightscout, "FetchGlucoseManager timer heartbeat")
  142. self.updateGlucoseSource()
  143. return self.glucoseSource.fetch(self.timer).eraseToAnyPublisher()
  144. }
  145. .sink { glucose in
  146. debug(.nightscout, "FetchGlucoseManager callback sensor")
  147. Publishers.CombineLatest3(
  148. Just(glucose),
  149. Just(self.glucoseStorage.syncDate()),
  150. self.healthKitManager.fetch(nil)
  151. )
  152. .eraseToAnyPublisher()
  153. .sink { newGlucose, syncDate, glucoseFromHealth in
  154. self.glucoseStoreAndHeartDecision(
  155. syncDate: syncDate,
  156. glucose: newGlucose,
  157. glucoseFromHealth: glucoseFromHealth
  158. )
  159. }
  160. .store(in: &self.lifetime)
  161. }
  162. .store(in: &lifetime)
  163. timer.fire()
  164. timer.resume()
  165. UserDefaults.standard
  166. .publisher(for: \.dexcomTransmitterID)
  167. .removeDuplicates()
  168. .sink { id in
  169. if self.settingsManager.settings.cgm == .dexcomG5 {
  170. if id != self.dexcomSourceG5.transmitterID {
  171. self.dexcomSourceG5 = DexcomSourceG5(glucoseStorage: self.glucoseStorage, glucoseManager: self)
  172. }
  173. } else if self.settingsManager.settings.cgm == .dexcomG6 {
  174. if id != self.dexcomSourceG6.transmitterID {
  175. self.dexcomSourceG6 = DexcomSourceG6(glucoseStorage: self.glucoseStorage, glucoseManager: self)
  176. }
  177. }
  178. }
  179. .store(in: &lifetime)
  180. }
  181. func sourceInfo() -> [String: Any]? {
  182. glucoseSource.sourceInfo()
  183. }
  184. }
  185. extension UserDefaults {
  186. @objc var dexcomTransmitterID: String? {
  187. get {
  188. string(forKey: "DexcomSource.transmitterID")?.nonEmpty
  189. }
  190. set {
  191. set(newValue, forKey: "DexcomSource.transmitterID")
  192. }
  193. }
  194. }