GlucoseStorage.swift 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. import Foundation
  2. import SwiftDate
  3. import Swinject
  4. protocol GlucoseStorage {
  5. func storeGlucose(_ glucose: [BloodGlucose])
  6. func recent() -> [BloodGlucose]
  7. func syncDate() -> Date
  8. func filterTooFrequentGlucose(_ glucose: [BloodGlucose], at: Date) -> [BloodGlucose]
  9. func lastGlucoseDate() -> Date
  10. func isGlucoseFresh() -> Bool
  11. func isGlucoseNotFlat() -> Bool
  12. func nightscoutGlucoseNotUploaded() -> [BloodGlucose]
  13. var alarm: GlucoseAlarm? { get }
  14. }
  15. final class BaseGlucoseStorage: GlucoseStorage, Injectable {
  16. private let processQueue = DispatchQueue(label: "BaseGlucoseStorage.processQueue")
  17. @Injected() private var storage: FileStorage!
  18. @Injected() private var broadcaster: Broadcaster!
  19. private enum Config {
  20. static let filterTime: TimeInterval = 4.5 * 60
  21. }
  22. init(resolver: Resolver) {
  23. injectServices(resolver)
  24. }
  25. func storeGlucose(_ glucose: [BloodGlucose]) {
  26. processQueue.sync {
  27. let file = OpenAPS.Monitor.glucose
  28. self.storage.transaction { storage in
  29. storage.append(glucose, to: file, uniqBy: \.dateString)
  30. let uniqEvents = storage.retrieve(file, as: [BloodGlucose].self)?
  31. .filter { $0.dateString.addingTimeInterval(1.days.timeInterval) > Date() }
  32. .sorted { $0.dateString > $1.dateString } ?? []
  33. let glucose = Array(uniqEvents)
  34. storage.save(glucose, as: file)
  35. DispatchQueue.main.async {
  36. self.broadcaster.notify(GlucoseObserver.self, on: .main) {
  37. $0.glucoseDidUpdate(glucose.reversed())
  38. }
  39. }
  40. }
  41. }
  42. }
  43. func syncDate() -> Date {
  44. guard let events = storage.retrieve(OpenAPS.Monitor.glucose, as: [BloodGlucose].self),
  45. let recent = events.first
  46. else {
  47. return Date().addingTimeInterval(-1.days.timeInterval)
  48. }
  49. return recent.dateString
  50. }
  51. func recent() -> [BloodGlucose] {
  52. storage.retrieve(OpenAPS.Monitor.glucose, as: [BloodGlucose].self)?.reversed() ?? []
  53. }
  54. func lastGlucoseDate() -> Date {
  55. recent().last?.dateString ?? .distantPast
  56. }
  57. func isGlucoseFresh() -> Bool {
  58. Date().timeIntervalSince(lastGlucoseDate()) <= Config.filterTime
  59. }
  60. func filterTooFrequentGlucose(_ glucose: [BloodGlucose], at date: Date) -> [BloodGlucose] {
  61. var lastDate = date
  62. var filtered: [BloodGlucose] = []
  63. for entry in glucose.reversed() {
  64. guard entry.dateString.addingTimeInterval(-Config.filterTime) > lastDate else {
  65. continue
  66. }
  67. filtered.append(entry)
  68. lastDate = entry.dateString
  69. }
  70. return filtered
  71. }
  72. func isGlucoseNotFlat() -> Bool {
  73. let last3 = recent().suffix(3)
  74. guard last3.count == 3 else { return true }
  75. return Array(
  76. last3
  77. .compactMap { $0.filtered ?? 0 }
  78. .filter { $0 != 0 }
  79. .uniqued()
  80. ).count != 1
  81. }
  82. func nightscoutGlucoseNotUploaded() -> [BloodGlucose] {
  83. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedGlucose, as: [BloodGlucose].self) ?? []
  84. let recentGlucose = recent()
  85. return Array(Set(recentGlucose).subtracting(Set(uploaded)))
  86. }
  87. var alarm: GlucoseAlarm? {
  88. guard let glucose = recent().last, glucose.dateString.addingTimeInterval(20.minutes.timeInterval) > Date(),
  89. let glucoseValue = glucose.glucose else { return nil }
  90. if glucoseValue < 72 {
  91. return .low
  92. }
  93. if glucoseValue > 270 {
  94. return .high
  95. }
  96. return nil
  97. }
  98. }
  99. protocol GlucoseObserver {
  100. func glucoseDidUpdate(_ glucose: [BloodGlucose])
  101. }
  102. enum GlucoseAlarm {
  103. case high
  104. case low
  105. var displayName: String {
  106. switch self {
  107. case .high:
  108. return NSLocalizedString("LOWALERT!", comment: "LOWALERT!")
  109. case .low:
  110. return NSLocalizedString("HIGHALERT!", comment: "HIGHALERT!")
  111. }
  112. }
  113. }