MockCGMManager.swift 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. //
  2. // MockCGMManager.swift
  3. // LoopKit
  4. //
  5. // Created by Michael Pangburn on 11/20/18.
  6. // Copyright © 2018 LoopKit Authors. All rights reserved.
  7. //
  8. import HealthKit
  9. import LoopKit
  10. import LoopTestingKit
  11. public struct MockCGMState: SensorDisplayable {
  12. public var isStateValid: Bool
  13. public var trendType: GlucoseTrend?
  14. public var isLocal: Bool {
  15. return true
  16. }
  17. }
  18. public final class MockCGMManager: TestingCGMManager {
  19. public static let managerIdentifier = "MockCGMManager"
  20. public static let localizedTitle = "Simulator"
  21. public var mockSensorState: MockCGMState {
  22. didSet {
  23. delegate.notify { (delegate) in
  24. delegate?.cgmManagerDidUpdateState(self)
  25. }
  26. }
  27. }
  28. public var sensorState: SensorDisplayable? {
  29. return mockSensorState
  30. }
  31. public var testingDevice: HKDevice {
  32. return MockCGMDataSource.device
  33. }
  34. public var device: HKDevice? {
  35. return testingDevice
  36. }
  37. public weak var cgmManagerDelegate: CGMManagerDelegate? {
  38. get {
  39. return delegate.delegate
  40. }
  41. set {
  42. delegate.delegate = newValue
  43. }
  44. }
  45. public var delegateQueue: DispatchQueue! {
  46. get {
  47. return delegate.queue
  48. }
  49. set {
  50. delegate.queue = newValue
  51. }
  52. }
  53. private let delegate = WeakSynchronizedDelegate<CGMManagerDelegate>()
  54. public var dataSource: MockCGMDataSource {
  55. didSet {
  56. delegate.notify { (delegate) in
  57. delegate?.cgmManagerDidUpdateState(self)
  58. }
  59. }
  60. }
  61. private var glucoseUpdateTimer: Timer?
  62. public init?(rawState: RawStateValue) {
  63. if let mockSensorStateRawValue = rawState["mockSensorState"] as? MockCGMState.RawValue,
  64. let mockSensorState = MockCGMState(rawValue: mockSensorStateRawValue) {
  65. self.mockSensorState = mockSensorState
  66. } else {
  67. self.mockSensorState = MockCGMState(isStateValid: true, trendType: nil)
  68. }
  69. if let dataSourceRawValue = rawState["dataSource"] as? MockCGMDataSource.RawValue,
  70. let dataSource = MockCGMDataSource(rawValue: dataSourceRawValue) {
  71. self.dataSource = dataSource
  72. } else {
  73. self.dataSource = MockCGMDataSource(model: .noData)
  74. }
  75. setupGlucoseUpdateTimer()
  76. }
  77. deinit {
  78. glucoseUpdateTimer?.invalidate()
  79. }
  80. public var rawState: RawStateValue {
  81. return [
  82. "mockSensorState": mockSensorState.rawValue,
  83. "dataSource": dataSource.rawValue
  84. ]
  85. }
  86. public let appURL: URL? = nil
  87. public let providesBLEHeartbeat = false
  88. public let managedDataInterval: TimeInterval? = nil
  89. public let shouldSyncToRemoteService = false
  90. public func fetchNewDataIfNeeded(_ completion: @escaping (CGMResult) -> Void) {
  91. dataSource.fetchNewData(completion)
  92. }
  93. public func backfillData(datingBack duration: TimeInterval) {
  94. let now = Date()
  95. dataSource.backfillData(from: DateInterval(start: now.addingTimeInterval(-duration), end: now)) { result in
  96. self.delegate.notify { delegate in
  97. delegate?.cgmManager(self, didUpdateWith: result)
  98. }
  99. }
  100. }
  101. private func setupGlucoseUpdateTimer() {
  102. glucoseUpdateTimer = Timer.scheduledTimer(withTimeInterval: dataSource.dataPointFrequency, repeats: true) { [weak self] _ in
  103. guard let self = self else { return }
  104. self.dataSource.fetchNewData { result in
  105. self.delegate.notify { delegate in
  106. delegate?.cgmManager(self, didUpdateWith: result)
  107. }
  108. }
  109. }
  110. }
  111. public func injectGlucoseSamples(_ samples: [NewGlucoseSample]) {
  112. guard !samples.isEmpty else { return }
  113. var samples = samples
  114. samples.mutateEach { $0.device = device }
  115. let result = CGMResult.newData(samples)
  116. delegate.notify { delegate in
  117. delegate?.cgmManager(self, didUpdateWith: result)
  118. }
  119. }
  120. }
  121. extension MockCGMManager {
  122. public var debugDescription: String {
  123. return """
  124. ## MockCGMManager
  125. state: \(mockSensorState)
  126. dataSource: \(dataSource)
  127. """
  128. }
  129. }
  130. extension MockCGMState: RawRepresentable {
  131. public typealias RawValue = [String: Any]
  132. public init?(rawValue: RawValue) {
  133. guard let isStateValid = rawValue["isStateValid"] as? Bool else {
  134. return nil
  135. }
  136. self.isStateValid = isStateValid
  137. if let trendTypeRawValue = rawValue["trendType"] as? GlucoseTrend.RawValue {
  138. self.trendType = GlucoseTrend(rawValue: trendTypeRawValue)
  139. }
  140. }
  141. public var rawValue: RawValue {
  142. var rawValue: RawValue = [
  143. "isStateValid": isStateValid,
  144. ]
  145. if let trendType = trendType {
  146. rawValue["trendType"] = trendType.rawValue
  147. }
  148. return rawValue
  149. }
  150. }
  151. extension MockCGMState: CustomDebugStringConvertible {
  152. public var debugDescription: String {
  153. return """
  154. ## MockCGMState
  155. * isStateValid: \(isStateValid)
  156. * trendType: \(trendType as Any)
  157. """
  158. }
  159. }