OverrideStorage.swift 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  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: @preconcurrency 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 0
  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.isfAndCr = override.isfAndCr
  109. newOverride.isf = override.isf
  110. newOverride.cr = override.cr
  111. newOverride.enabled = override.enabled
  112. newOverride.smbIsOff = override.smbIsOff
  113. if override.overrideTarget {
  114. newOverride.target = override.target as NSDecimalNumber
  115. } else {
  116. newOverride.target = 0
  117. }
  118. if override.advancedSettings {
  119. newOverride.advancedSettings = true
  120. newOverride.smbMinutes = override.smbMinutes as NSDecimalNumber
  121. newOverride.uamMinutes = override.uamMinutes as NSDecimalNumber
  122. }
  123. if override.smbIsScheduledOff {
  124. newOverride.smbIsScheduledOff = true
  125. newOverride.start = override.start as NSDecimalNumber
  126. newOverride.end = override.end as NSDecimalNumber
  127. } else {
  128. newOverride.smbIsScheduledOff = false
  129. }
  130. do {
  131. guard self.backgroundContext.hasChanges else { return }
  132. try self.backgroundContext.save()
  133. } catch let error as NSError {
  134. debugPrint(
  135. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to save Override Preset to Core Data with error: \(error.userInfo)"
  136. )
  137. }
  138. }
  139. }
  140. // Copy the current Override if it is a RUNNING Preset
  141. /// otherwise we would edit the Preset
  142. @MainActor func copyRunningOverride(_ override: OverrideStored) async -> NSManagedObjectID {
  143. let newOverride = OverrideStored(context: viewContext)
  144. newOverride.duration = override.duration
  145. newOverride.indefinite = override.indefinite
  146. newOverride.percentage = override.percentage
  147. newOverride.smbIsOff = override.smbIsOff
  148. newOverride.name = override.name
  149. newOverride.isPreset = false // no Preset
  150. newOverride.date = override.date
  151. newOverride.enabled = override.enabled
  152. newOverride.target = override.target
  153. newOverride.advancedSettings = override.advancedSettings
  154. newOverride.isfAndCr = override.isfAndCr
  155. newOverride.isf = override.isf
  156. newOverride.cr = override.cr
  157. newOverride.smbIsScheduledOff = override.smbIsScheduledOff
  158. newOverride.start = override.start
  159. newOverride.end = override.end
  160. newOverride.smbMinutes = override.smbMinutes
  161. newOverride.uamMinutes = override.uamMinutes
  162. newOverride.isUploadedToNS = true // set to true to avoid getting duplicate entries on NS
  163. await viewContext.perform {
  164. do {
  165. guard self.viewContext.hasChanges else { return }
  166. try self.viewContext.save()
  167. } catch let error as NSError {
  168. debugPrint(
  169. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to copy Override with error: \(error.userInfo)"
  170. )
  171. }
  172. }
  173. return newOverride.objectID
  174. }
  175. /// marked as MainActor to be able to publish changes from the background
  176. /// - Parameter: NSManagedObjectID to be able to transfer the object safely from one thread to another thread
  177. @MainActor func deleteOverridePreset(_ objectID: NSManagedObjectID) async {
  178. await CoreDataStack.shared.deleteObject(identifiedBy: objectID)
  179. }
  180. func getOverridesNotYetUploadedToNightscout() async -> [NightscoutExercise] {
  181. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  182. ofType: OverrideStored.self,
  183. onContext: backgroundContext,
  184. predicate: NSPredicate.lastActiveOverrideNotYetUploadedToNightscout,
  185. key: "date",
  186. ascending: false
  187. )
  188. return await backgroundContext.perform {
  189. guard let fetchedOverrides = results as? [OverrideStored] else { return [] }
  190. return fetchedOverrides.map { override in
  191. let duration = override.indefinite ? 1440 : override.duration ?? 0 // 1440 min = 1 day
  192. return NightscoutExercise(
  193. duration: Int(truncating: duration),
  194. eventType: OverrideStored.EventType.nsExercise,
  195. createdAt: override.date ?? Date(),
  196. enteredBy: NightscoutExercise.local,
  197. notes: override.name ?? "Custom Override",
  198. id: UUID(uuidString: override.id ?? UUID().uuidString)
  199. )
  200. }
  201. }
  202. }
  203. func getOverrideRunsNotYetUploadedToNightscout() async -> [NightscoutExercise] {
  204. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  205. ofType: OverrideRunStored.self,
  206. onContext: backgroundContext,
  207. predicate: NSPredicate(
  208. format: "startDate >= %@ AND isUploadedToNS == %@",
  209. Date.oneDayAgo as NSDate,
  210. false as NSNumber
  211. ),
  212. key: "startDate",
  213. ascending: false
  214. )
  215. return await backgroundContext.perform {
  216. guard let fetchedOverrideRuns = results as? [OverrideRunStored] else { return [] }
  217. return fetchedOverrideRuns.map { overrideRun in
  218. var durationInMinutes = (overrideRun.endDate?.timeIntervalSince(overrideRun.startDate ?? Date()) ?? 1) / 60
  219. durationInMinutes = durationInMinutes < 1 ? 1 : durationInMinutes
  220. return NightscoutExercise(
  221. duration: Int(durationInMinutes),
  222. eventType: OverrideStored.EventType.nsExercise,
  223. createdAt: (overrideRun.startDate ?? overrideRun.override?.date) ?? Date(),
  224. enteredBy: NightscoutExercise.local,
  225. notes: overrideRun.name ?? "Custom Override",
  226. id: overrideRun.id
  227. )
  228. }
  229. }
  230. }
  231. func getPresetOverridesForNightscout() async -> [NightscoutPresetOverride] {
  232. let results = await CoreDataStack.shared.fetchEntitiesAsync(
  233. ofType: OverrideStored.self,
  234. onContext: backgroundContext,
  235. predicate: NSPredicate.allOverridePresets,
  236. key: "orderPosition",
  237. ascending: true
  238. )
  239. return await backgroundContext.perform {
  240. guard let fetchedResults = results as? [OverrideStored] else { return [] }
  241. return fetchedResults.map { overrideStored in
  242. let duration = overrideStored.duration as? Decimal != 0 ? overrideStored.duration as? Decimal : nil
  243. let percentage = overrideStored.percentage != 0 ? overrideStored.percentage : nil
  244. let target = (overrideStored.target as? Decimal) != 0 ? overrideStored.target as? Decimal : nil
  245. return NightscoutPresetOverride(
  246. name: overrideStored.name ?? "",
  247. duration: duration,
  248. percentage: percentage,
  249. target: target
  250. )
  251. }
  252. }
  253. }
  254. }