TransmitterManager.swift 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. //
  2. // TransmitterManager.swift
  3. // Loop
  4. //
  5. // Copyright © 2017 LoopKit Authors. All rights reserved.
  6. //
  7. import os.log
  8. import HealthKit
  9. public protocol TransmitterManagerDelegate: AnyObject {
  10. func transmitterManager(_ manager: TransmitterManager, didRead glucose: [Glucose])
  11. func transmitterManager(_ manager: TransmitterManager, didFailWith error: Error)
  12. }
  13. public struct TransmitterManagerState: Equatable {
  14. public static let version = 1
  15. public var transmitterID: String
  16. public var passiveModeEnabled: Bool = true
  17. public var shouldSyncToRemoteService: Bool
  18. public init(transmitterID: String, shouldSyncToRemoteService: Bool = false) {
  19. self.transmitterID = transmitterID
  20. self.shouldSyncToRemoteService = shouldSyncToRemoteService
  21. }
  22. }
  23. public protocol TransmitterManagerObserver: AnyObject {
  24. func transmitterManagerDidUpdateLatestReading(_ manager: TransmitterManager)
  25. }
  26. public class TransmitterManager: TransmitterDelegate {
  27. public weak var delegate: TransmitterManagerDelegate?
  28. private var state: TransmitterManagerState
  29. private let observers = WeakSynchronizedSet<TransmitterManagerObserver>()
  30. public var hasValidSensorSession: Bool {
  31. // TODO: we should decode and persist transmitter session state
  32. return !state.transmitterID.isEmpty
  33. }
  34. public required init(state: TransmitterManagerState) {
  35. self.state = state
  36. self.transmitter = Transmitter(id: state.transmitterID, passiveModeEnabled: state.passiveModeEnabled)
  37. self.transmitter.delegate = self
  38. }
  39. public var shouldSyncToRemoteService: Bool {
  40. get {
  41. return state.shouldSyncToRemoteService
  42. }
  43. set {
  44. self.state.shouldSyncToRemoteService = newValue
  45. notifyDelegateOfStateChange()
  46. }
  47. }
  48. private(set) public var latestConnection: Date? {
  49. get {
  50. return lockedLatestConnection.value
  51. }
  52. set {
  53. lockedLatestConnection.value = newValue
  54. }
  55. }
  56. private let lockedLatestConnection: Locked<Date?> = Locked(nil)
  57. public let transmitter: Transmitter
  58. let log = OSLog(category: "TransmitterManager")
  59. public var providesBLEHeartbeat: Bool {
  60. return dataIsFresh
  61. }
  62. private(set) public var latestReading: Glucose? {
  63. get {
  64. return lockedLatestReading.value
  65. }
  66. set {
  67. lockedLatestReading.value = newValue
  68. }
  69. }
  70. private let lockedLatestReading: Locked<Glucose?> = Locked(nil)
  71. private var dataIsFresh: Bool {
  72. guard let latestGlucose = latestReading,
  73. latestGlucose.readDate > Date(timeIntervalSinceNow: .minutes(-4.5)) else {
  74. return false
  75. }
  76. return true
  77. }
  78. // public func fetchNewDataIfNeeded(_ completion: @escaping (CGMReadingResult) -> Void) {
  79. // // Ensure our transmitter connection is active
  80. // transmitter.resumeScanning()
  81. //
  82. // // If our last glucose was less than 4.5 minutes ago, don't fetch.
  83. // guard !dataIsFresh else {
  84. // completion(.noData)
  85. // return
  86. // }
  87. //
  88. // log.default("Fetching new glucose from Share because last reading is %{public}.1f minutes old", latestReading?.readDate.timeIntervalSinceNow.minutes ?? 0)
  89. // }
  90. public var debugDescription: String {
  91. return [
  92. "## \(String(describing: type(of: self)))",
  93. "latestReading: \(String(describing: latestReading))",
  94. "latestConnection: \(String(describing: latestConnection))",
  95. "dataIsFresh: \(dataIsFresh)",
  96. "providesBLEHeartbeat: \(providesBLEHeartbeat)",
  97. "observers.count: \(observers.cleanupDeallocatedElements().count)",
  98. String(reflecting: transmitter),
  99. ].joined(separator: "\n")
  100. }
  101. // private func updateDelegate(with result: CGMReadingResult) {
  102. // if let manager = self as? CGMManager {
  103. // shareManager.delegate.notify { (delegate) in
  104. // delegate?.cgmManager(manager, hasNew: result)
  105. // }
  106. // }
  107. //
  108. // notifyObserversOfLatestReading()
  109. // }
  110. private func notifyDelegateOfStateChange() {
  111. // if let manager = self as? CGMManager {
  112. // shareManager.delegate.notify { (delegate) in
  113. // delegate?.cgmManagerDidUpdateState(manager)
  114. // }
  115. // }
  116. }
  117. // MARK: - TransmitterDelegate
  118. public func transmitterDidConnect(_ transmitter: Transmitter) {
  119. log.default("%{public}@", #function)
  120. latestConnection = Date()
  121. // logDeviceCommunication("Connected", type: .connection)
  122. }
  123. public func transmitter(_ transmitter: Transmitter, didError error: Error) {
  124. log.error("%{public}@: %{public}@", #function, String(describing: error))
  125. // updateDelegate(with: .error(error))
  126. // logDeviceCommunication("Error: \(error)", type: .error)
  127. }
  128. public func transmitter(_ transmitter: Transmitter, didRead glucose: Glucose) {
  129. guard glucose != latestReading else {
  130. delegate?.transmitterManager(self, didRead: [])
  131. // updateDelegate(with: .noData)
  132. return
  133. }
  134. latestReading = glucose
  135. // logDeviceCommunication("New reading: \(glucose.readDate)", type: .receive)
  136. guard glucose.state.hasReliableGlucose else {
  137. log.default("%{public}@: Unreliable glucose: %{public}@", #function, String(describing: glucose.state))
  138. delegate?.transmitterManager(self, didFailWith: CalibrationError.unreliableState(glucose.state))
  139. // updateDelegate(with: .error(CalibrationError.unreliableState(glucose.state)))
  140. return
  141. }
  142. guard glucose.glucose != nil else {
  143. delegate?.transmitterManager(self, didRead: [])
  144. // updateDelegate(with: .noData)
  145. return
  146. }
  147. log.default("%{public}@: New glucose", #function)
  148. delegate?.transmitterManager(self, didRead: [glucose])
  149. // updateDelegate(with: .newData([
  150. // NewGlucoseSample(
  151. // date: glucose.readDate,
  152. // quantity: quantity,
  153. // trend: glucose.trendType,
  154. // isDisplayOnly: glucose.isDisplayOnly,
  155. // wasUserEntered: glucose.isDisplayOnly,
  156. // syncIdentifier: glucose.syncIdentifier,
  157. // device: device
  158. // )
  159. // ]))
  160. }
  161. public func transmitter(_ transmitter: Transmitter, didReadBackfill glucose: [Glucose]) {
  162. let samples = glucose.filter { glucose -> Bool in
  163. guard glucose != latestReading, glucose.state.hasReliableGlucose, glucose.glucose != nil else {
  164. return false
  165. }
  166. return true
  167. }
  168. delegate?.transmitterManager(self, didRead: samples)
  169. //
  170. // guard samples.count > 0 else {
  171. // return
  172. // }
  173. // updateDelegate(with: .newData(samples))
  174. // logDeviceCommunication("New backfill: \(String(describing: samples.first?.date))", type: .receive)
  175. }
  176. public func transmitter(_ transmitter: Transmitter, didReadUnknownData data: Data) {
  177. log.error("Unknown sensor data: %{public}@", data.hexadecimalString)
  178. // This can be used for protocol discovery, but isn't necessary for normal operation
  179. // logDeviceCommunication("Unknown sensor data: \(data.hexadecimalString)", type: .error)
  180. }
  181. }
  182. // MARK: - Observer management
  183. extension TransmitterManager {
  184. public func addObserver(_ observer: TransmitterManagerObserver, queue: DispatchQueue) {
  185. observers.insert(observer, queue: queue)
  186. }
  187. public func removeObserver(_ observer: TransmitterManagerObserver) {
  188. observers.removeElement(observer)
  189. }
  190. private func notifyObserversOfLatestReading() {
  191. observers.forEach { (observer) in
  192. observer.transmitterManagerDidUpdateLatestReading(self)
  193. }
  194. }
  195. }
  196. public class G5CGMManager: TransmitterManager {
  197. public let managerIdentifier: String = "DexG5Transmitter"
  198. public let localizedTitle = LocalizedString("Dexcom G5", comment: "CGM display title")
  199. public let isOnboarded = true // No distinction between created and onboarded
  200. public var appURL: URL? {
  201. return URL(string: "dexcomcgm://")
  202. }
  203. public var device: HKDevice? {
  204. return HKDevice(
  205. name: "CGMBLEKit",
  206. manufacturer: "Dexcom",
  207. model: "G5 Mobile",
  208. hardwareVersion: nil,
  209. firmwareVersion: nil,
  210. softwareVersion: String(CGMBLEKitVersionNumber),
  211. localIdentifier: nil,
  212. udiDeviceIdentifier: "00386270000002"
  213. )
  214. }
  215. // func logDeviceCommunication(_ message: String, type: DeviceLogEntryType = .send) {
  216. // self.cgmManagerDelegate?.deviceManager(self, logEventForDeviceIdentifier: transmitter.ID, type: type, message: message, completion: nil)
  217. // }
  218. }
  219. public class G6CGMManager: TransmitterManager {
  220. public let managerIdentifier: String = "DexG6Transmitter"
  221. public let localizedTitle = LocalizedString("Dexcom G6", comment: "CGM display title")
  222. public let isOnboarded = true // No distinction between created and onboarded
  223. public var appURL: URL? {
  224. return nil
  225. }
  226. public var device: HKDevice? {
  227. return HKDevice(
  228. name: "CGMBLEKit",
  229. manufacturer: "Dexcom",
  230. model: "G6",
  231. hardwareVersion: nil,
  232. firmwareVersion: nil,
  233. softwareVersion: String(CGMBLEKitVersionNumber),
  234. localIdentifier: nil,
  235. udiDeviceIdentifier: "00386270000385"
  236. )
  237. }
  238. // func logDeviceCommunication(_ message: String, type: DeviceLogEntryType = .send) {
  239. // self.cgmManagerDelegate?.deviceManager(self, logEventForDeviceIdentifier: transmitter.ID, type: type, message: message, completion: nil)
  240. // }
  241. }
  242. enum CalibrationError: Error {
  243. case unreliableState(CalibrationState)
  244. }
  245. extension CalibrationError: LocalizedError {
  246. var errorDescription: String? {
  247. switch self {
  248. case .unreliableState:
  249. return LocalizedString("Glucose data is unavailable", comment: "Error description for unreliable state")
  250. }
  251. }
  252. var failureReason: String? {
  253. switch self {
  254. case .unreliableState(let state):
  255. return state.localizedDescription
  256. }
  257. }
  258. }
  259. extension CalibrationState {
  260. public var localizedDescription: String {
  261. switch self {
  262. case .known(let state):
  263. switch state {
  264. case .needCalibration7, .needCalibration14, .needFirstInitialCalibration, .needSecondInitialCalibration, .calibrationError8, .calibrationError9, .calibrationError10, .calibrationError13:
  265. return LocalizedString("Sensor needs calibration", comment: "The description of sensor calibration state when sensor needs calibration.")
  266. case .ok:
  267. return LocalizedString("Sensor calibration is OK", comment: "The description of sensor calibration state when sensor calibration is ok.")
  268. case .stopped, .sensorFailure11, .sensorFailure12, .sessionFailure15, .sessionFailure16, .sessionFailure17:
  269. return LocalizedString("Sensor is stopped", comment: "The description of sensor calibration state when sensor sensor is stopped.")
  270. case .warmup, .questionMarks:
  271. return LocalizedString("Sensor is warming up", comment: "The description of sensor calibration state when sensor sensor is warming up.")
  272. }
  273. case .unknown(let rawValue):
  274. return String(format: LocalizedString("Sensor is in unknown state %1$d", comment: "The description of sensor calibration state when raw value is unknown. (1: missing data details)"), rawValue)
  275. }
  276. }
  277. }