OverrideStorage.swift 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. import CoreData
  2. import Foundation
  3. import Swinject
  4. protocol OverrideStorage {
  5. func fetchLastCreatedOverride() async -> [NSManagedObjectID]
  6. func loadLatestOverrideConfigurations(fetchLimit: Int) async -> [NSManagedObjectID]
  7. func fetchForOverridePresets() async -> [NSManagedObjectID]
  8. func calculateTarget(override: OverrideStored) -> Decimal
  9. func storeOverride(override: Override) async
  10. func copyRunningOverride(_ override: OverrideStored) async -> NSManagedObjectID
  11. func deleteOverridePreset(_ objectID: NSManagedObjectID) async
  12. func getOverridesNotYetUploadedToNightscout() async -> [NightscoutExercise]
  13. func getOverrideRunsNotYetUploadedToNightscout() async -> [NightscoutExercise]
  14. func getPresetOverridesForNightscout() async -> [NightscoutPresetOverride]
  15. }
  16. final class BaseOverrideStorage: OverrideStorage, Injectable {
  17. @Injected() private var settingsManager: SettingsManager!
  18. private let viewContext = CoreDataStack.shared.persistentContainer.viewContext
  19. private let backgroundContext = CoreDataStack.shared.newTaskContext()
  20. init(resolver: Resolver) {
  21. injectServices(resolver)
  22. }
  23. private var dateFormatter: DateFormatter {
  24. let dateFormatter = DateFormatter()
  25. dateFormatter.dateStyle = .short
  26. dateFormatter.timeStyle = .short
  27. dateFormatter.locale = Locale.current
  28. return dateFormatter
  29. }
  30. func fetchLastCreatedOverride() async -> [NSManagedObjectID] {
  31. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  32. ofType: OverrideStored.self,
  33. onContext: backgroundContext,
  34. predicate: NSPredicate(
  35. format: "date >= %@",
  36. Date.oneDayAgo as NSDate
  37. ),
  38. key: "date",
  39. ascending: false,
  40. fetchLimit: 1
  41. )
  42. return await backgroundContext.perform {
  43. guard let fetchedResults = results as? [OverrideStored] else { return [] }
  44. return fetchedResults.map(\.objectID)
  45. }
  46. }
  47. func loadLatestOverrideConfigurations(fetchLimit: Int) async -> [NSManagedObjectID] {
  48. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  49. ofType: OverrideStored.self,
  50. onContext: backgroundContext,
  51. predicate: NSPredicate.lastActiveOverride,
  52. key: "orderPosition",
  53. ascending: true,
  54. fetchLimit: fetchLimit
  55. )
  56. return await backgroundContext.perform {
  57. guard let fetchedResults = results as? [OverrideStored] else { return [] }
  58. return fetchedResults.map(\.objectID)
  59. }
  60. }
  61. /// Returns the NSManagedObjectID of the Override Presets
  62. func fetchForOverridePresets() async -> [NSManagedObjectID] {
  63. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  64. ofType: OverrideStored.self,
  65. onContext: backgroundContext,
  66. predicate: NSPredicate.allOverridePresets,
  67. key: "orderPosition",
  68. ascending: true
  69. )
  70. return await backgroundContext.perform {
  71. guard let fetchedResults = results as? [OverrideStored] else { return [] }
  72. return fetchedResults.map(\.objectID)
  73. }
  74. }
  75. @MainActor func calculateTarget(override: OverrideStored) -> Decimal {
  76. guard let overrideTarget = override.target, overrideTarget != 0 else {
  77. return 100 // default
  78. }
  79. return overrideTarget.decimalValue
  80. }
  81. func storeOverride(override: Override) async {
  82. var presetCount = -1
  83. if override.isPreset {
  84. let presets = await fetchForOverridePresets()
  85. presetCount = presets.count
  86. }
  87. await backgroundContext.perform {
  88. let newOverride = OverrideStored(context: self.backgroundContext)
  89. // override key meta data
  90. if !override.name.isEmpty {
  91. newOverride.name = override.name
  92. } else {
  93. let formattedDate = self.dateFormatter.string(from: Date())
  94. newOverride.name = "Override \(formattedDate)"
  95. }
  96. newOverride.id = UUID().uuidString
  97. newOverride.date = override.date
  98. newOverride.isPreset = override.isPreset
  99. newOverride.isUploadedToNS = false
  100. // Assign orderPosition if it's a preset and presetCount is valid
  101. if override.isPreset, presetCount > -1 {
  102. newOverride.orderPosition = Int16(presetCount + 1) // Ensure type matches Core Data model
  103. }
  104. // override metrics
  105. newOverride.duration = override.duration as NSDecimalNumber
  106. newOverride.indefinite = override.indefinite
  107. newOverride.percentage = override.percentage
  108. newOverride.enabled = override.enabled
  109. newOverride.smbIsOff = override.smbIsOff
  110. if override.overrideTarget {
  111. newOverride.target = (
  112. self.settingsManager.settings.units == .mmolL ? override.target.asMgdL : override.target
  113. ) as NSDecimalNumber
  114. } else {
  115. newOverride.target = 0
  116. }
  117. if override.advancedSettings {
  118. newOverride.advancedSettings = true
  119. if !override.isfAndCr {
  120. newOverride.isfAndCr = false
  121. newOverride.isf = override.isf
  122. newOverride.cr = override.cr
  123. } else {
  124. newOverride.isfAndCr = true
  125. }
  126. if override.smbIsAlwaysOff {
  127. newOverride.smbIsAlwaysOff = true
  128. newOverride.start = override.start as NSDecimalNumber
  129. newOverride.end = override.end as NSDecimalNumber
  130. } else {
  131. newOverride.smbIsAlwaysOff = false
  132. }
  133. newOverride.smbMinutes = override.smbMinutes as NSDecimalNumber
  134. newOverride.uamMinutes = override.uamMinutes as NSDecimalNumber
  135. }
  136. do {
  137. guard self.backgroundContext.hasChanges else { return }
  138. try self.backgroundContext.save()
  139. } catch let error as NSError {
  140. debugPrint(
  141. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to save Override Preset to Core Data with error: \(error.userInfo)"
  142. )
  143. }
  144. }
  145. }
  146. // Copy the current Override if it is a RUNNING Preset
  147. /// otherwise we would edit the Preset
  148. @MainActor func copyRunningOverride(_ override: OverrideStored) async -> NSManagedObjectID {
  149. let newOverride = OverrideStored(context: viewContext)
  150. newOverride.duration = override.duration
  151. newOverride.indefinite = override.indefinite
  152. newOverride.percentage = override.percentage
  153. newOverride.smbIsOff = override.smbIsOff
  154. newOverride.name = override.name
  155. newOverride.isPreset = false // no Preset
  156. newOverride.date = override.date
  157. newOverride.enabled = override.enabled
  158. newOverride.target = override.target
  159. newOverride.advancedSettings = override.advancedSettings
  160. newOverride.isfAndCr = override.isfAndCr
  161. newOverride.isf = override.isf
  162. newOverride.cr = override.cr
  163. newOverride.smbIsAlwaysOff = override.smbIsAlwaysOff
  164. newOverride.start = override.start
  165. newOverride.end = override.end
  166. newOverride.smbMinutes = override.smbMinutes
  167. newOverride.uamMinutes = override.uamMinutes
  168. newOverride.isUploadedToNS = true // set to true to avoid getting duplicate entries on NS
  169. await viewContext.perform {
  170. do {
  171. guard self.viewContext.hasChanges else { return }
  172. try self.viewContext.save()
  173. } catch let error as NSError {
  174. debugPrint(
  175. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to copy Override with error: \(error.userInfo)"
  176. )
  177. }
  178. }
  179. return newOverride.objectID
  180. }
  181. /// marked as MainActor to be able to publish changes from the background
  182. /// - Parameter: NSManagedObjectID to be able to transfer the object safely from one thread to another thread
  183. @MainActor func deleteOverridePreset(_ objectID: NSManagedObjectID) async {
  184. await CoreDataStack.shared.deleteObject(identifiedBy: objectID)
  185. }
  186. func getOverridesNotYetUploadedToNightscout() async -> [NightscoutExercise] {
  187. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  188. ofType: OverrideStored.self,
  189. onContext: backgroundContext,
  190. predicate: NSPredicate.lastActiveOverrideNotYetUploadedToNightscout,
  191. key: "date",
  192. ascending: false
  193. )
  194. return await backgroundContext.perform {
  195. guard let fetchedOverrides = results as? [OverrideStored] else { return [] }
  196. return fetchedOverrides.map { override in
  197. let duration = override.indefinite ? 1440 : override.duration ?? 0 // 1440 min = 1 day
  198. return NightscoutExercise(
  199. duration: Int(truncating: duration),
  200. eventType: OverrideStored.EventType.nsExercise,
  201. createdAt: override.date ?? Date(),
  202. enteredBy: NightscoutExercise.local,
  203. notes: override.name ?? "Custom Override",
  204. id: UUID(uuidString: override.id ?? UUID().uuidString)
  205. )
  206. }
  207. }
  208. }
  209. func getOverrideRunsNotYetUploadedToNightscout() async -> [NightscoutExercise] {
  210. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  211. ofType: OverrideRunStored.self,
  212. onContext: backgroundContext,
  213. predicate: NSPredicate(
  214. format: "startDate >= %@ AND isUploadedToNS == %@",
  215. Date.oneDayAgo as NSDate,
  216. false as NSNumber
  217. ),
  218. key: "startDate",
  219. ascending: false
  220. )
  221. return await backgroundContext.perform {
  222. guard let fetchedOverrideRuns = results as? [OverrideRunStored] else { return [] }
  223. return fetchedOverrideRuns.map { overrideRun in
  224. var durationInMinutes = (overrideRun.endDate?.timeIntervalSince(overrideRun.startDate ?? Date()) ?? 1) / 60
  225. durationInMinutes = durationInMinutes < 1 ? 1 : durationInMinutes
  226. return NightscoutExercise(
  227. duration: Int(durationInMinutes),
  228. eventType: OverrideStored.EventType.nsExercise,
  229. createdAt: (overrideRun.startDate ?? overrideRun.override?.date) ?? Date(),
  230. enteredBy: NightscoutExercise.local,
  231. notes: overrideRun.name ?? "Custom Override",
  232. id: overrideRun.id
  233. )
  234. }
  235. }
  236. }
  237. func getPresetOverridesForNightscout() async -> [NightscoutPresetOverride] {
  238. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  239. ofType: OverrideStored.self,
  240. onContext: backgroundContext,
  241. predicate: NSPredicate.allOverridePresets,
  242. key: "orderPosition",
  243. ascending: true
  244. )
  245. return await backgroundContext.perform {
  246. guard let fetchedResults = results as? [OverrideStored] else { return [] }
  247. return fetchedResults.map { overrideStored in
  248. let duration = overrideStored.duration as? Decimal != 0 ? overrideStored.duration as? Decimal : nil
  249. let percentage = overrideStored.percentage != 0 ? overrideStored.percentage : nil
  250. let target = (overrideStored.target as? Decimal) != 0 ? overrideStored.target as? Decimal : nil
  251. return NightscoutPresetOverride(
  252. name: overrideStored.name ?? "",
  253. duration: duration,
  254. percentage: percentage,
  255. target: target
  256. )
  257. }
  258. }
  259. }
  260. }