TempTargetsStorage.swift 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  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 fetchScheduledTempTargets() async -> [NSManagedObjectID]
  13. func fetchScheduledTempTarget(for targetDate: Date) async -> [NSManagedObjectID]
  14. func copyRunningTempTarget(_ tempTarget: TempTargetStored) async -> NSManagedObjectID
  15. func deleteOverridePreset(_ objectID: NSManagedObjectID) async
  16. func loadLatestTempTargetConfigurations(fetchLimit: Int) async -> [NSManagedObjectID]
  17. func syncDate() -> Date
  18. func recent() -> [TempTarget]
  19. func nightscoutTreatmentsNotUploaded() -> [NightscoutTreatment]
  20. func presets() -> [TempTarget]
  21. func current() -> TempTarget?
  22. }
  23. final class BaseTempTargetsStorage: TempTargetsStorage, Injectable {
  24. private let processQueue = DispatchQueue(label: "BaseTempTargetsStorage.processQueue")
  25. @Injected() private var storage: FileStorage!
  26. @Injected() private var broadcaster: Broadcaster!
  27. private let backgroundContext = CoreDataStack.shared.newTaskContext()
  28. private let viewContext = CoreDataStack.shared.persistentContainer.viewContext
  29. init(resolver: Resolver) {
  30. injectServices(resolver)
  31. }
  32. func loadLatestTempTargetConfigurations(fetchLimit: Int) async -> [NSManagedObjectID] {
  33. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  34. ofType: TempTargetStored.self,
  35. onContext: backgroundContext,
  36. predicate: NSPredicate.lastActiveTempTarget,
  37. key: "orderPosition",
  38. ascending: true,
  39. fetchLimit: fetchLimit
  40. )
  41. guard let fetchedResults = results as? [TempTargetStored] else { return [] }
  42. return await backgroundContext.perform {
  43. return fetchedResults.map(\.objectID)
  44. }
  45. }
  46. /// Returns the NSManagedObjectID of the Temp Target Presets
  47. func fetchForTempTargetPresets() async -> [NSManagedObjectID] {
  48. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  49. ofType: TempTargetStored.self,
  50. onContext: backgroundContext,
  51. predicate: NSPredicate.allTempTargetPresets,
  52. key: "orderPosition",
  53. ascending: true
  54. )
  55. guard let fetchedResults = results as? [TempTargetStored] else { return [] }
  56. return await backgroundContext.perform {
  57. return fetchedResults.map(\.objectID)
  58. }
  59. }
  60. func fetchScheduledTempTargets() async -> [NSManagedObjectID] {
  61. let scheduledTempTargets = NSPredicate(format: "date > %@", Date() as NSDate)
  62. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  63. ofType: TempTargetStored.self,
  64. onContext: backgroundContext,
  65. predicate: scheduledTempTargets,
  66. key: "date",
  67. ascending: false
  68. )
  69. guard let fetchedResults = results as? [TempTargetStored] else { return [] }
  70. return await backgroundContext.perform {
  71. return fetchedResults.map(\.objectID)
  72. }
  73. }
  74. func fetchScheduledTempTarget(for targetDate: Date) async -> [NSManagedObjectID] {
  75. let predicate = NSPredicate(format: "date == %@", targetDate as NSDate)
  76. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  77. ofType: TempTargetStored.self,
  78. onContext: backgroundContext,
  79. predicate: predicate,
  80. key: "date",
  81. ascending: false,
  82. fetchLimit: 1
  83. )
  84. guard let fetchedResults = results as? [TempTargetStored] else { return [] }
  85. return await backgroundContext.perform {
  86. fetchedResults.map(\.objectID)
  87. }
  88. }
  89. func storeTempTarget(tempTarget: TempTarget) async {
  90. var presetCount = -1
  91. if tempTarget.isPreset == true {
  92. let presets = await fetchForTempTargetPresets()
  93. presetCount = presets.count
  94. }
  95. await backgroundContext.perform {
  96. let newTempTarget = TempTargetStored(context: self.backgroundContext)
  97. newTempTarget.date = tempTarget.createdAt
  98. newTempTarget.id = UUID()
  99. newTempTarget.enabled = tempTarget.enabled ?? false
  100. newTempTarget.duration = tempTarget.duration as NSDecimalNumber
  101. newTempTarget.isUploadedToNS = false
  102. newTempTarget.name = tempTarget.name
  103. newTempTarget.target = NSDecimalNumber(decimal: tempTarget.targetTop ?? 0)
  104. newTempTarget.isPreset = tempTarget.isPreset ?? false
  105. newTempTarget.halfBasalTarget = NSDecimalNumber(decimal: tempTarget.halfBasalTarget ?? 160)
  106. // Set order position if we have a valid count and the temp target is a preset
  107. if tempTarget.isPreset == true, presetCount > -1 {
  108. newTempTarget.orderPosition = Int16(presetCount + 1)
  109. }
  110. do {
  111. guard self.backgroundContext.hasChanges else { return }
  112. try self.backgroundContext.save()
  113. } catch let error as NSError {
  114. debugPrint(
  115. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to save Temp Target to Core Data with error: \(error.userInfo)"
  116. )
  117. }
  118. }
  119. }
  120. func saveTempTargetsToStorage(_ targets: [TempTarget]) {
  121. processQueue.async {
  122. let file = OpenAPS.Settings.tempTargets
  123. var uniqEvents: [TempTarget] = []
  124. self.storage.transaction { storage in
  125. storage.append(targets, to: file, uniqBy: \.createdAt)
  126. let retrievedTargets = storage.retrieve(file, as: [TempTarget].self) ?? []
  127. uniqEvents = retrievedTargets
  128. .filter { $0.isWithinLastDay }
  129. .sorted(by: { $0.createdAt > $1.createdAt })
  130. storage.save(uniqEvents, as: file)
  131. }
  132. self.broadcaster.notify(TempTargetsObserver.self, on: self.processQueue) {
  133. $0.tempTargetsDidUpdate(uniqEvents)
  134. }
  135. }
  136. }
  137. // Copy the current Temp Target if it is a RUNNING Preset
  138. /// otherwise we would edit the Preset
  139. @MainActor func copyRunningTempTarget(_ tempTarget: TempTargetStored) async -> NSManagedObjectID {
  140. let newTempTarget = TempTargetStored(context: viewContext)
  141. newTempTarget.date = tempTarget.date
  142. newTempTarget.id = tempTarget.id
  143. newTempTarget.enabled = tempTarget.enabled
  144. newTempTarget.duration = tempTarget.duration
  145. newTempTarget.isUploadedToNS = true // to avoid getting duplicates on NS
  146. newTempTarget.name = tempTarget.name
  147. newTempTarget.target = tempTarget.target
  148. newTempTarget.isPreset = false // no Preset
  149. newTempTarget.halfBasalTarget = tempTarget.halfBasalTarget
  150. await viewContext.perform {
  151. do {
  152. guard self.viewContext.hasChanges else { return }
  153. try self.viewContext.save()
  154. } catch let error as NSError {
  155. debugPrint(
  156. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to copy Temp Target with error: \(error.userInfo)"
  157. )
  158. }
  159. }
  160. return newTempTarget.objectID
  161. }
  162. @MainActor func deleteOverridePreset(_ objectID: NSManagedObjectID) async {
  163. await CoreDataStack.shared.deleteObject(identifiedBy: objectID)
  164. }
  165. func syncDate() -> Date {
  166. Date().addingTimeInterval(-1.days.timeInterval)
  167. }
  168. func recent() -> [TempTarget] {
  169. storage.retrieve(OpenAPS.Settings.tempTargets, as: [TempTarget].self)?.reversed() ?? []
  170. }
  171. func current() -> TempTarget? {
  172. guard let last = recent().last else {
  173. return nil
  174. }
  175. guard last.createdAt.addingTimeInterval(Int(last.duration).minutes.timeInterval) > Date(), last.createdAt <= Date(),
  176. last.duration != 0
  177. else {
  178. return nil
  179. }
  180. return last
  181. }
  182. func nightscoutTreatmentsNotUploaded() -> [NightscoutTreatment] {
  183. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedTempTargets, as: [NightscoutTreatment].self) ?? []
  184. let eventsManual = recent().filter { $0.enteredBy == TempTarget.manual }
  185. let treatments = eventsManual.map {
  186. NightscoutTreatment(
  187. duration: Int($0.duration),
  188. rawDuration: nil,
  189. rawRate: nil,
  190. absolute: nil,
  191. rate: nil,
  192. eventType: .nsTempTarget,
  193. createdAt: $0.createdAt,
  194. enteredBy: TempTarget.manual,
  195. bolus: nil,
  196. insulin: nil,
  197. notes: nil,
  198. carbs: nil,
  199. targetTop: $0.targetTop,
  200. targetBottom: $0.targetBottom
  201. )
  202. }
  203. return Array(Set(treatments).subtracting(Set(uploaded)))
  204. }
  205. func presets() -> [TempTarget] {
  206. storage.retrieve(OpenAPS.FreeAPS.tempTargetsPresets, as: [TempTarget].self)?.reversed() ?? []
  207. }
  208. }
  209. private extension TempTarget {
  210. var isActive: Bool {
  211. let expirationTime = createdAt.addingTimeInterval(Int(duration).minutes.timeInterval)
  212. return expirationTime > Date() && createdAt <= Date()
  213. }
  214. var isWithinLastDay: Bool {
  215. createdAt.addingTimeInterval(1.days.timeInterval) > Date()
  216. }
  217. }