GlucoseStored+helper.swift 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. import CoreData
  2. import Foundation
  3. extension GlucoseStored {
  4. static func fetch(
  5. _ predicate: NSPredicate = .all,
  6. ascending: Bool,
  7. fetchLimit: Int? = nil,
  8. batchSize: Int? = nil
  9. ) -> NSFetchRequest<GlucoseStored> {
  10. let request = GlucoseStored.fetchRequest()
  11. request.sortDescriptors = [NSSortDescriptor(keyPath: \GlucoseStored.date, ascending: ascending)]
  12. request.predicate = predicate
  13. if let limit = fetchLimit {
  14. request.fetchLimit = limit
  15. }
  16. if let batchSize = batchSize {
  17. request.fetchBatchSize = batchSize
  18. }
  19. return request
  20. }
  21. static func glucoseIsHIGH(_ glucose: [GlucoseStored]) -> Bool {
  22. guard glucose.count >= 4 else { return false }
  23. let firstValue = glucose.first?.glucose
  24. /// 400 mg/dL covers all Dexcom CGMs as well as European Libre 2 and most readings from xDrip4iOS.
  25. /// U.S. / Canadian Libres can emit up to 500 mg/dL until it reads "HI"
  26. /// Our condition considers both these values, 400 and 500, as possible "flat" readings when paired CGM reads HIGH.
  27. return glucose.allSatisfy { $0.glucose == firstValue && ($0.glucose == 400 || $0.glucose == 500) }
  28. }
  29. // Preview
  30. @discardableResult static func makePreviewGlucose(count: Int, provider: CoreDataStack) -> [GlucoseStored] {
  31. let context = provider.persistentContainer.viewContext
  32. let baseGlucose = 120
  33. let glucoseValues = (0 ..< count).map { index -> GlucoseStored in
  34. let glucose = GlucoseStored(context: context)
  35. glucose.id = UUID()
  36. glucose.date = Date.now.addingTimeInterval(Double(index) * -300) // Every 5 minutes
  37. glucose.glucose = Int16(baseGlucose + (index % 3) * 10) // Varying between 120-140
  38. glucose.direction = BloodGlucose.Direction.flat.rawValue
  39. glucose.isManual = false
  40. glucose.isUploadedToNS = false
  41. glucose.isUploadedToHealth = false
  42. glucose.isUploadedToTidepool = false
  43. return glucose
  44. }
  45. try? context.save()
  46. return glucoseValues
  47. }
  48. }
  49. extension NSPredicate {
  50. static var glucose: NSPredicate {
  51. let date = Date.oneDayAgo
  52. return NSPredicate(format: "date >= %@", date as NSDate)
  53. }
  54. static var manualGlucose: NSPredicate {
  55. let date = Date.oneDayAgo
  56. return NSPredicate(format: "isManual == %@ AND date >= %@", true as NSNumber, date as NSDate)
  57. }
  58. static var glucoseForStatsDay: NSPredicate {
  59. let date = Date.oneDayAgo
  60. return NSPredicate(format: "date >= %@", date as NSDate)
  61. }
  62. static var glucoseForStatsToday: NSPredicate {
  63. let date = Date.startOfToday
  64. return NSPredicate(format: "date >= %@", date as NSDate)
  65. }
  66. static var glucoseForStatsMonth: NSPredicate {
  67. let date = Date.oneMonthAgo
  68. return NSPredicate(format: "date >= %@", date as NSDate)
  69. }
  70. static var glucoseForStatsTotal: NSPredicate {
  71. let date = Date.threeMonthsAgo
  72. return NSPredicate(format: "date >= %@", date as NSDate)
  73. }
  74. static var glucoseForStatsWeek: NSPredicate {
  75. let date = Date.oneWeekAgo
  76. return NSPredicate(format: "date >= %@", date as NSDate)
  77. }
  78. static var glucoseNotYetUploadedToNightscout: NSPredicate {
  79. let date = Date.oneDayAgo
  80. return NSPredicate(format: "date >= %@ AND isUploadedToNS == %@", date as NSDate, false as NSNumber)
  81. }
  82. static var glucoseNotYetUploadedToHealth: NSPredicate {
  83. let date = Date.oneDayAgo
  84. return NSPredicate(format: "date >= %@ AND isUploadedToHealth == %@", date as NSDate, false as NSNumber)
  85. }
  86. static var glucoseNotYetUploadedToTidepool: NSPredicate {
  87. let date = Date.oneDayAgo
  88. return NSPredicate(format: "date >= %@ AND isUploadedToTidepool == %@", date as NSDate, false as NSNumber)
  89. }
  90. static var manualGlucoseNotYetUploadedToNightscout: NSPredicate {
  91. let date = Date.oneDayAgo
  92. return NSPredicate(
  93. format: "date >= %@ AND isUploadedToNS == %@ AND isManual == %@",
  94. date as NSDate,
  95. false as NSNumber,
  96. true as NSNumber
  97. )
  98. }
  99. static var manualGlucoseNotYetUploadedToHealth: NSPredicate {
  100. let date = Date.oneDayAgo
  101. return NSPredicate(
  102. format: "date >= %@ AND isUploadedToHealth == %@ AND isManual == %@",
  103. date as NSDate,
  104. false as NSNumber,
  105. true as NSNumber
  106. )
  107. }
  108. static var manualGlucoseNotYetUploadedToTidepool: NSPredicate {
  109. let date = Date.oneDayAgo
  110. return NSPredicate(
  111. format: "date >= %@ AND isUploadedToTidepool == %@ AND isManual == %@",
  112. date as NSDate,
  113. false as NSNumber,
  114. true as NSNumber
  115. )
  116. }
  117. }
  118. extension GlucoseStored: Encodable {
  119. enum CodingKeys: String, CodingKey {
  120. case date
  121. case dateString
  122. case sgv
  123. case glucose
  124. case direction
  125. case id
  126. case type
  127. }
  128. public func encode(to encoder: Encoder) throws {
  129. var container = encoder.container(keyedBy: CodingKeys.self)
  130. let dateFormatter = ISO8601DateFormatter()
  131. dateFormatter.formatOptions = [.withInternetDateTime, .withFractionalSeconds]
  132. try container.encode(dateFormatter.string(from: date ?? Date()), forKey: .dateString)
  133. let dateAsUnixTimestamp = String(format: "%.0f", (date?.timeIntervalSince1970 ?? Date().timeIntervalSince1970) * 1000)
  134. try container.encode(dateAsUnixTimestamp, forKey: .date)
  135. try container.encode(direction, forKey: .direction)
  136. try container.encode(id, forKey: .id)
  137. // TODO: Handle the type of the glucose entry conditionally not hardcoded
  138. try container.encode("sgv", forKey: .type)
  139. if isManual {
  140. try container.encode(glucose, forKey: .glucose)
  141. } else {
  142. try container.encode(glucose, forKey: .sgv)
  143. }
  144. }
  145. }
  146. // In order to show the correct direction in the bobble we convert the direction property of the NSManagedObject GlucoseStored back to the Direction type
  147. extension GlucoseStored {
  148. var directionEnum: BloodGlucose.Direction? {
  149. BloodGlucose.Direction(rawValue: direction ?? "")
  150. }
  151. }