DexcomSource.swift 2.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. import CGMBLEKit
  2. import Combine
  3. import Foundation
  4. final class DexcomSource: GlucoseSource {
  5. private let processQueue = DispatchQueue(label: "DexcomSource.processQueue")
  6. private let dexcomManager = TransmitterManager(
  7. state: TransmitterManagerState(transmitterID: UserDefaults.standard.dexcomTransmitterID ?? "000000")
  8. )
  9. private var promise: Future<[BloodGlucose], Error>.Promise?
  10. init() {
  11. dexcomManager.delegate = self
  12. }
  13. var transmitterID: String {
  14. dexcomManager.transmitter.ID
  15. }
  16. func fetch() -> AnyPublisher<[BloodGlucose], Never> {
  17. dexcomManager.transmitter.resumeScanning()
  18. return Future<[BloodGlucose], Error> { [weak self] promise in
  19. self?.promise = promise
  20. }
  21. .timeout(60, scheduler: processQueue, options: nil, customError: nil)
  22. .replaceError(with: [])
  23. .eraseToAnyPublisher()
  24. }
  25. deinit {
  26. dexcomManager.transmitter.stopScanning()
  27. }
  28. }
  29. extension DexcomSource: TransmitterManagerDelegate {
  30. func transmitterManager(_: TransmitterManager, didFailWith error: Error) {
  31. promise?(.failure(error))
  32. }
  33. func transmitterManager(_: TransmitterManager, didRead glucose: [CGMBLEKit.Glucose]) {
  34. let bloodGlucose = glucose.compactMap { glucose -> BloodGlucose? in
  35. guard let quantity = glucose.glucose else {
  36. return nil
  37. }
  38. let value = Int(quantity.doubleValue(for: .milligramsPerDeciliter))
  39. return BloodGlucose(
  40. _id: glucose.syncIdentifier,
  41. sgv: value,
  42. direction: .init(trend: glucose.trend),
  43. date: Decimal(Int(glucose.readDate.timeIntervalSince1970 * 1000)),
  44. dateString: glucose.readDate,
  45. unfiltered: nil,
  46. filtered: nil,
  47. noise: nil,
  48. glucose: value,
  49. type: "sgv"
  50. )
  51. }
  52. promise?(.success(bloodGlucose))
  53. }
  54. }
  55. extension BloodGlucose.Direction {
  56. init(trend: Int) {
  57. guard trend < Int(Int8.max) else {
  58. self = .none
  59. return
  60. }
  61. switch trend {
  62. case let x where x <= -30:
  63. self = .doubleDown
  64. case let x where x <= -20:
  65. self = .singleDown
  66. case let x where x <= -10:
  67. self = .fortyFiveDown
  68. case let x where x < 10:
  69. self = .flat
  70. case let x where x < 20:
  71. self = .fortyFiveUp
  72. case let x where x < 30:
  73. self = .singleUp
  74. default:
  75. self = .doubleUp
  76. }
  77. }
  78. }