FetchGlucoseManager.swift 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. import Combine
  2. import Foundation
  3. import SwiftDate
  4. import Swinject
  5. protocol FetchGlucoseManager: SourceInfoProvider {}
  6. final class BaseFetchGlucoseManager: FetchGlucoseManager, Injectable {
  7. private let processQueue = DispatchQueue(label: "BaseGlucoseManager.processQueue")
  8. @Injected() var glucoseStorage: GlucoseStorage!
  9. @Injected() var nightscoutManager: NightscoutManager!
  10. @Injected() var apsManager: APSManager!
  11. @Injected() var settingsManager: SettingsManager!
  12. @Injected() var libreTransmitter: LibreTransmitterSource!
  13. @Injected() var healthKitManager: HealthKitManager!
  14. private var lifetime = Lifetime()
  15. private let timer = DispatchTimer(timeInterval: 1.minutes.timeInterval)
  16. private lazy var appGroupSource = AppGroupSource()
  17. private lazy var dexcomSource = DexcomSource()
  18. private lazy var simulatorSource = GlucoseSimulatorSource()
  19. init(resolver: Resolver) {
  20. injectServices(resolver)
  21. updateGlucoseSource()
  22. subscribe()
  23. }
  24. var glucoseSource: GlucoseSource!
  25. private func updateGlucoseSource() {
  26. switch settingsManager.settings.cgm {
  27. case .xdrip:
  28. glucoseSource = appGroupSource
  29. case .dexcomG5,
  30. .dexcomG6:
  31. glucoseSource = dexcomSource
  32. case .nightscout:
  33. glucoseSource = nightscoutManager
  34. case .simulator:
  35. glucoseSource = simulatorSource
  36. case .libreTransmitter:
  37. glucoseSource = libreTransmitter
  38. }
  39. if settingsManager.settings.cgm != .libreTransmitter {
  40. libreTransmitter.manager = nil
  41. }
  42. }
  43. private func subscribe() {
  44. timer.publisher
  45. .receive(on: processQueue)
  46. .flatMap { date -> AnyPublisher<(Date, Date, [BloodGlucose]), Never> in
  47. debug(.nightscout, "FetchGlucoseManager heartbeat")
  48. debug(.nightscout, "Start fetching glucose")
  49. self.updateGlucoseSource()
  50. return Publishers.CombineLatest3(
  51. Just(date),
  52. Just(self.glucoseStorage.syncDate()),
  53. self.glucoseSource.fetch().merge(with: self.healthKitManager.fetch())
  54. .eraseToAnyPublisher()
  55. )
  56. .eraseToAnyPublisher()
  57. }
  58. .sink { date, syncDate, glucose in
  59. // Because of Spike dosn't respect a date query
  60. let filteredByDate = glucose.filter { $0.dateString > syncDate }
  61. let filtered = self.glucoseStorage.filterTooFrequentGlucose(filteredByDate, at: syncDate)
  62. if !filtered.isEmpty {
  63. debug(.nightscout, "New glucose found")
  64. self.glucoseStorage.storeGlucose(filtered)
  65. self.apsManager.heartbeat(date: date, force: false)
  66. self.nightscoutManager.uploadGlucose()
  67. self.healthKitManager.save(bloodGlucoses: filtered, completion: nil)
  68. }
  69. }
  70. .store(in: &lifetime)
  71. timer.fire()
  72. timer.resume()
  73. UserDefaults.standard
  74. .publisher(for: \.dexcomTransmitterID)
  75. .removeDuplicates()
  76. .sink { id in
  77. guard [.dexcomG5, .dexcomG6].contains(self.settingsManager.settings.cgm) else { return }
  78. if id != self.dexcomSource.transmitterID {
  79. self.dexcomSource = DexcomSource()
  80. }
  81. }
  82. .store(in: &lifetime)
  83. }
  84. func sourceInfo() -> [String: Any]? {
  85. glucoseSource.sourceInfo()
  86. }
  87. }
  88. extension UserDefaults {
  89. @objc var dexcomTransmitterID: String? {
  90. get {
  91. string(forKey: "DexcomSource.transmitterID")?.nonEmpty
  92. }
  93. set {
  94. set(newValue, forKey: "DexcomSource.transmitterID")
  95. }
  96. }
  97. }