OverrideStorage.swift 12 KB

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