TempTargetsStorage.swift 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  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 fetchForTempTargetPresets() async -> [NSManagedObjectID]
  11. func copyRunningTempTarget(_ tempTarget: TempTargetStored) async -> NSManagedObjectID
  12. func deleteOverridePreset(_ objectID: NSManagedObjectID) async
  13. func loadLatestTempTargetConfigurations(fetchLimit: Int) async -> [NSManagedObjectID]
  14. func syncDate() -> Date
  15. func recent() -> [TempTarget]
  16. func nightscoutTreatmentsNotUploaded() -> [NightscoutTreatment]
  17. // func storePresets(_ targets: [TempTarget])
  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. do {
  70. guard self.backgroundContext.hasChanges else { return }
  71. try self.backgroundContext.save()
  72. } catch let error as NSError {
  73. debugPrint(
  74. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to save Temp Target to Core Data with error: \(error.userInfo)"
  75. )
  76. }
  77. if tempTarget.isPreset ?? false {
  78. self.saveTempTargetsToStorage([tempTarget], isPreset: true)
  79. } else {
  80. self.saveTempTargetsToStorage([tempTarget], isPreset: false)
  81. }
  82. }
  83. }
  84. private func saveTempTargetsToStorage(_ targets: [TempTarget], isPreset: Bool) {
  85. processQueue.sync {
  86. var updatedTargets = targets
  87. if !isPreset, let currentTarget = current() {
  88. if let newActive = updatedTargets.last(where: { $0.isActive }) {
  89. // Cancel current target
  90. updatedTargets.append(.cancel(at: newActive.createdAt.addingTimeInterval(-1)))
  91. }
  92. }
  93. let file = isPreset ? OpenAPS.FreeAPS.tempTargetsPresets : OpenAPS.Settings.tempTargets
  94. var uniqEvents: [TempTarget] = []
  95. self.storage.transaction { storage in
  96. storage.append(updatedTargets, to: file, uniqBy: \.createdAt)
  97. let retrievedTargets = storage.retrieve(file, as: [TempTarget].self) ?? []
  98. uniqEvents = retrievedTargets
  99. .filter { isPreset || $0.isWithinLastDay }
  100. .sorted(by: { $0.createdAt > $1.createdAt })
  101. storage.save(uniqEvents, as: file)
  102. }
  103. broadcaster.notify(TempTargetsObserver.self, on: processQueue) {
  104. $0.tempTargetsDidUpdate(uniqEvents)
  105. }
  106. }
  107. }
  108. // Copy the current Temp Target if it is a RUNNING Preset
  109. /// otherwise we would edit the Preset
  110. @MainActor func copyRunningTempTarget(_ tempTarget: TempTargetStored) async -> NSManagedObjectID {
  111. let newTempTarget = TempTargetStored(context: viewContext)
  112. newTempTarget.date = tempTarget.date
  113. newTempTarget.id = tempTarget.id
  114. newTempTarget.enabled = tempTarget.enabled
  115. newTempTarget.duration = tempTarget.duration
  116. newTempTarget.isUploadedToNS = true // to avoid getting duplicates on NS
  117. newTempTarget.name = tempTarget.name
  118. newTempTarget.target = tempTarget.target
  119. newTempTarget.isPreset = false // no Preset
  120. await viewContext.perform {
  121. do {
  122. guard self.viewContext.hasChanges else { return }
  123. try self.viewContext.save()
  124. } catch let error as NSError {
  125. debugPrint(
  126. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to copy Temp Target with error: \(error.userInfo)"
  127. )
  128. }
  129. }
  130. return newTempTarget.objectID
  131. }
  132. @MainActor func deleteOverridePreset(_ objectID: NSManagedObjectID) async {
  133. await CoreDataStack.shared.deleteObject(identifiedBy: objectID)
  134. }
  135. func syncDate() -> Date {
  136. Date().addingTimeInterval(-1.days.timeInterval)
  137. }
  138. func recent() -> [TempTarget] {
  139. storage.retrieve(OpenAPS.Settings.tempTargets, as: [TempTarget].self)?.reversed() ?? []
  140. }
  141. func current() -> TempTarget? {
  142. guard let last = recent().last else {
  143. return nil
  144. }
  145. guard last.createdAt.addingTimeInterval(Int(last.duration).minutes.timeInterval) > Date(), last.createdAt <= Date(),
  146. last.duration != 0
  147. else {
  148. return nil
  149. }
  150. return last
  151. }
  152. func nightscoutTreatmentsNotUploaded() -> [NightscoutTreatment] {
  153. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedTempTargets, as: [NightscoutTreatment].self) ?? []
  154. let eventsManual = recent().filter { $0.enteredBy == TempTarget.manual }
  155. let treatments = eventsManual.map {
  156. NightscoutTreatment(
  157. duration: Int($0.duration),
  158. rawDuration: nil,
  159. rawRate: nil,
  160. absolute: nil,
  161. rate: nil,
  162. eventType: .nsTempTarget,
  163. createdAt: $0.createdAt,
  164. enteredBy: TempTarget.manual,
  165. bolus: nil,
  166. insulin: nil,
  167. notes: nil,
  168. carbs: nil,
  169. targetTop: $0.targetTop,
  170. targetBottom: $0.targetBottom
  171. )
  172. }
  173. return Array(Set(treatments).subtracting(Set(uploaded)))
  174. }
  175. // func storePresets(_ targets: [TempTarget]) {
  176. // storage.remove(OpenAPS.FreeAPS.tempTargetsPresets)
  177. //
  178. // saveTempTargetsToStorage(targets, isPreset: true)
  179. // }
  180. //
  181. func presets() -> [TempTarget] {
  182. storage.retrieve(OpenAPS.FreeAPS.tempTargetsPresets, as: [TempTarget].self)?.reversed() ?? []
  183. }
  184. }
  185. private extension TempTarget {
  186. var isActive: Bool {
  187. let expirationTime = createdAt.addingTimeInterval(Int(duration).minutes.timeInterval)
  188. return expirationTime > Date() && createdAt <= Date()
  189. }
  190. var isWithinLastDay: Bool {
  191. createdAt.addingTimeInterval(1.days.timeInterval) > Date()
  192. }
  193. }