FetchGlucoseManager.swift 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  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. case .glucoseDirect:
  39. glucoseSource = appGroupSource
  40. }
  41. if settingsManager.settings.cgm != .libreTransmitter {
  42. libreTransmitter.manager = nil
  43. }
  44. }
  45. private func subscribe() {
  46. timer.publisher
  47. .receive(on: processQueue)
  48. .flatMap { date -> AnyPublisher<(Date, Date, [BloodGlucose]), Never> in
  49. debug(.nightscout, "FetchGlucoseManager heartbeat")
  50. debug(.nightscout, "Start fetching glucose")
  51. self.updateGlucoseSource()
  52. return Publishers.CombineLatest3(
  53. Just(date),
  54. Just(self.glucoseStorage.syncDate()),
  55. self.glucoseSource.fetch()
  56. )
  57. .eraseToAnyPublisher()
  58. }
  59. .sink { date, syncDate, glucose in
  60. // Because of Spike dosn't respect a date query
  61. let filteredByDate = glucose.filter { $0.dateString > syncDate }
  62. let filtered = self.glucoseStorage.filterTooFrequentGlucose(filteredByDate, at: syncDate)
  63. if !filtered.isEmpty {
  64. debug(.nightscout, "New glucose found")
  65. self.glucoseStorage.storeGlucose(filtered)
  66. self.apsManager.heartbeat(date: date, force: false)
  67. self.nightscoutManager.uploadGlucose()
  68. self.healthKitManager.save(bloodGlucoses: filtered, completion: nil)
  69. }
  70. }
  71. .store(in: &lifetime)
  72. timer.fire()
  73. timer.resume()
  74. UserDefaults.standard
  75. .publisher(for: \.dexcomTransmitterID)
  76. .removeDuplicates()
  77. .sink { id in
  78. guard [.dexcomG5, .dexcomG6].contains(self.settingsManager.settings.cgm) else { return }
  79. if id != self.dexcomSource.transmitterID {
  80. self.dexcomSource = DexcomSource()
  81. }
  82. }
  83. .store(in: &lifetime)
  84. }
  85. func sourceInfo() -> [String: Any]? {
  86. glucoseSource.sourceInfo()
  87. }
  88. }
  89. extension UserDefaults {
  90. @objc var dexcomTransmitterID: String? {
  91. get {
  92. string(forKey: "DexcomSource.transmitterID")?.nonEmpty
  93. }
  94. set {
  95. set(newValue, forKey: "DexcomSource.transmitterID")
  96. }
  97. }
  98. }