TempTargetsStorage.swift 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. import CoreData
  2. import Foundation
  3. import SwiftDate
  4. import Swinject
  5. protocol TempTargetsObserver {
  6. func tempTargetsDidUpdate(_ targets: [TempTarget])
  7. }
  8. protocol TempTargetsStorage {
  9. func storeTempTarget(tempTarget: TempTarget) async
  10. func saveTempTargetsToStorage(_ targets: [TempTarget])
  11. func fetchForTempTargetPresets() async -> [NSManagedObjectID]
  12. func copyRunningTempTarget(_ tempTarget: TempTargetStored) async -> NSManagedObjectID
  13. func deleteOverridePreset(_ objectID: NSManagedObjectID) async
  14. func loadLatestTempTargetConfigurations(fetchLimit: Int) async -> [NSManagedObjectID]
  15. func syncDate() -> Date
  16. func recent() -> [TempTarget]
  17. func nightscoutTreatmentsNotUploaded() -> [NightscoutTreatment]
  18. func presets() -> [TempTarget]
  19. func current() -> TempTarget?
  20. }
  21. final class BaseTempTargetsStorage: TempTargetsStorage, Injectable {
  22. private let processQueue = DispatchQueue(label: "BaseTempTargetsStorage.processQueue")
  23. @Injected() private var storage: FileStorage!
  24. @Injected() private var broadcaster: Broadcaster!
  25. private let backgroundContext = CoreDataStack.shared.newTaskContext()
  26. private let viewContext = CoreDataStack.shared.persistentContainer.viewContext
  27. init(resolver: Resolver) {
  28. injectServices(resolver)
  29. }
  30. func loadLatestTempTargetConfigurations(fetchLimit: Int) async -> [NSManagedObjectID] {
  31. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  32. ofType: TempTargetStored.self,
  33. onContext: backgroundContext,
  34. predicate: NSPredicate.lastActiveTempTarget,
  35. key: "date",
  36. ascending: true,
  37. fetchLimit: fetchLimit
  38. )
  39. guard let fetchedResults = results as? [TempTargetStored] else { return [] }
  40. return await backgroundContext.perform {
  41. return fetchedResults.map(\.objectID)
  42. }
  43. }
  44. /// Returns the NSManagedObjectID of the Temp Target Presets
  45. func fetchForTempTargetPresets() async -> [NSManagedObjectID] {
  46. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  47. ofType: TempTargetStored.self,
  48. onContext: backgroundContext,
  49. predicate: NSPredicate.allTempTargetPresets,
  50. key: "date",
  51. ascending: true
  52. )
  53. guard let fetchedResults = results as? [TempTargetStored] else { return [] }
  54. return await backgroundContext.perform {
  55. return fetchedResults.map(\.objectID)
  56. }
  57. }
  58. func storeTempTarget(tempTarget: TempTarget) async {
  59. await backgroundContext.perform {
  60. let newTempTarget = TempTargetStored(context: self.backgroundContext)
  61. newTempTarget.date = tempTarget.createdAt
  62. newTempTarget.id = UUID()
  63. newTempTarget.enabled = tempTarget.enabled ?? false
  64. newTempTarget.duration = tempTarget.duration as NSDecimalNumber
  65. newTempTarget.isUploadedToNS = false
  66. newTempTarget.name = tempTarget.name
  67. newTempTarget.target = NSDecimalNumber(decimal: tempTarget.targetTop ?? 0)
  68. newTempTarget.isPreset = tempTarget.isPreset ?? false
  69. newTempTarget.halfBasalTarget = NSDecimalNumber(decimal: tempTarget.halfBasalTarget ?? 160)
  70. do {
  71. guard self.backgroundContext.hasChanges else { return }
  72. try self.backgroundContext.save()
  73. } catch let error as NSError {
  74. debugPrint(
  75. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to save Temp Target to Core Data with error: \(error.userInfo)"
  76. )
  77. }
  78. /*
  79. Saving the Preset to the Storage means that it gets used by Oref
  80. We only want that when either creating a new non-Preset-Temp Target or when enacting a Temp Target Preset, NOT when we are only saving a new Preset, hence the check here!
  81. */
  82. if !(tempTarget.isPreset ?? false) {
  83. self.saveTempTargetsToStorage([tempTarget])
  84. }
  85. }
  86. }
  87. func saveTempTargetsToStorage(_ targets: [TempTarget]) {
  88. processQueue.async {
  89. var updatedTargets = targets
  90. if let newActive = updatedTargets.last(where: { $0.isActive }) {
  91. // Cancel current target
  92. // This logic to add a new cancel Temp target to cancel the current one is so fucking dumb...
  93. updatedTargets.append(.cancel(at: newActive.createdAt.addingTimeInterval(-1)))
  94. }
  95. let file = OpenAPS.Settings.tempTargets
  96. var uniqEvents: [TempTarget] = []
  97. self.storage.transaction { storage in
  98. storage.append(updatedTargets, to: file, uniqBy: \.createdAt)
  99. let retrievedTargets = storage.retrieve(file, as: [TempTarget].self) ?? []
  100. uniqEvents = retrievedTargets
  101. .filter { $0.isWithinLastDay }
  102. .sorted(by: { $0.createdAt > $1.createdAt })
  103. storage.save(uniqEvents, as: file)
  104. }
  105. self.broadcaster.notify(TempTargetsObserver.self, on: self.processQueue) {
  106. $0.tempTargetsDidUpdate(uniqEvents)
  107. }
  108. }
  109. }
  110. // Copy the current Temp Target if it is a RUNNING Preset
  111. /// otherwise we would edit the Preset
  112. @MainActor func copyRunningTempTarget(_ tempTarget: TempTargetStored) async -> NSManagedObjectID {
  113. let newTempTarget = TempTargetStored(context: viewContext)
  114. newTempTarget.date = tempTarget.date
  115. newTempTarget.id = tempTarget.id
  116. newTempTarget.enabled = tempTarget.enabled
  117. newTempTarget.duration = tempTarget.duration
  118. newTempTarget.isUploadedToNS = true // to avoid getting duplicates on NS
  119. newTempTarget.name = tempTarget.name
  120. newTempTarget.target = tempTarget.target
  121. newTempTarget.isPreset = false // no Preset
  122. newTempTarget.halfBasalTarget = tempTarget.halfBasalTarget
  123. await viewContext.perform {
  124. do {
  125. guard self.viewContext.hasChanges else { return }
  126. try self.viewContext.save()
  127. } catch let error as NSError {
  128. debugPrint(
  129. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to copy Temp Target with error: \(error.userInfo)"
  130. )
  131. }
  132. }
  133. return newTempTarget.objectID
  134. }
  135. @MainActor func deleteOverridePreset(_ objectID: NSManagedObjectID) async {
  136. await CoreDataStack.shared.deleteObject(identifiedBy: objectID)
  137. }
  138. func syncDate() -> Date {
  139. Date().addingTimeInterval(-1.days.timeInterval)
  140. }
  141. func recent() -> [TempTarget] {
  142. storage.retrieve(OpenAPS.Settings.tempTargets, as: [TempTarget].self)?.reversed() ?? []
  143. }
  144. func current() -> TempTarget? {
  145. guard let last = recent().last else {
  146. return nil
  147. }
  148. guard last.createdAt.addingTimeInterval(Int(last.duration).minutes.timeInterval) > Date(), last.createdAt <= Date(),
  149. last.duration != 0
  150. else {
  151. return nil
  152. }
  153. return last
  154. }
  155. func nightscoutTreatmentsNotUploaded() -> [NightscoutTreatment] {
  156. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedTempTargets, as: [NightscoutTreatment].self) ?? []
  157. let eventsManual = recent().filter { $0.enteredBy == TempTarget.manual }
  158. let treatments = eventsManual.map {
  159. NightscoutTreatment(
  160. duration: Int($0.duration),
  161. rawDuration: nil,
  162. rawRate: nil,
  163. absolute: nil,
  164. rate: nil,
  165. eventType: .nsTempTarget,
  166. createdAt: $0.createdAt,
  167. enteredBy: TempTarget.manual,
  168. bolus: nil,
  169. insulin: nil,
  170. notes: nil,
  171. carbs: nil,
  172. targetTop: $0.targetTop,
  173. targetBottom: $0.targetBottom
  174. )
  175. }
  176. return Array(Set(treatments).subtracting(Set(uploaded)))
  177. }
  178. func presets() -> [TempTarget] {
  179. storage.retrieve(OpenAPS.FreeAPS.tempTargetsPresets, as: [TempTarget].self)?.reversed() ?? []
  180. }
  181. }
  182. private extension TempTarget {
  183. var isActive: Bool {
  184. let expirationTime = createdAt.addingTimeInterval(Int(duration).minutes.timeInterval)
  185. return expirationTime > Date() && createdAt <= Date()
  186. }
  187. var isWithinLastDay: Bool {
  188. createdAt.addingTimeInterval(1.days.timeInterval) > Date()
  189. }
  190. }