OverrideStorage.swift 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406
  1. import CoreData
  2. import Foundation
  3. import Swinject
  4. protocol OverrideStorage {
  5. func fetchLastCreatedOverride() async throws -> [NSManagedObjectID]
  6. func loadLatestOverrideConfigurations(fetchLimit: Int) async throws -> [NSManagedObjectID]
  7. func fetchForOverridePresets() async throws -> [NSManagedObjectID]
  8. func calculateTarget(override: OverrideStored) -> Decimal
  9. func storeOverride(override: Override) async throws
  10. func copyRunningOverride(_ override: OverrideStored) async -> NSManagedObjectID
  11. func deleteOverridePreset(_ objectID: NSManagedObjectID) async
  12. func getOverridesNotYetUploadedToNightscout() async throws -> [NightscoutExercise]
  13. func getOverrideRunsNotYetUploadedToNightscout() async throws -> [NightscoutExercise]
  14. func checkIfShouldDeleteNightscoutOverrideEntry(
  15. forCreatedAt createdAtString: String,
  16. newDuration: Int?,
  17. using nightscout: NightscoutAPI
  18. ) async throws
  19. func getPresetOverridesForNightscout() async throws -> [NightscoutPresetOverride]
  20. func fetchLatestActiveOverride() async throws -> NSManagedObjectID?
  21. }
  22. final class BaseOverrideStorage: @preconcurrency OverrideStorage, Injectable {
  23. @Injected() private var settingsManager: SettingsManager!
  24. private let viewContext = CoreDataStack.shared.persistentContainer.viewContext
  25. private let context: NSManagedObjectContext
  26. init(resolver: Resolver, context: NSManagedObjectContext? = nil) {
  27. self.context = context ?? CoreDataStack.shared.newTaskContext()
  28. injectServices(resolver)
  29. }
  30. private var dateFormatter: DateFormatter {
  31. let dateFormatter = DateFormatter()
  32. dateFormatter.dateStyle = .short
  33. dateFormatter.timeStyle = .short
  34. dateFormatter.locale = Locale.current
  35. return dateFormatter
  36. }
  37. func fetchLastCreatedOverride() async throws -> [NSManagedObjectID] {
  38. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  39. ofType: OverrideStored.self,
  40. onContext: context,
  41. predicate: NSPredicate(
  42. format: "date >= %@",
  43. Date.oneDayAgo as NSDate
  44. ),
  45. key: "date",
  46. ascending: false,
  47. fetchLimit: 1
  48. )
  49. return try await context.perform {
  50. guard let fetchedResults = results as? [OverrideStored] else {
  51. throw CoreDataError.fetchError(function: #function, file: #file)
  52. }
  53. return fetchedResults.map(\.objectID)
  54. }
  55. }
  56. func loadLatestOverrideConfigurations(fetchLimit: Int) async throws -> [NSManagedObjectID] {
  57. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  58. ofType: OverrideStored.self,
  59. onContext: context,
  60. predicate: NSPredicate.lastActiveOverride,
  61. key: "orderPosition",
  62. ascending: true,
  63. fetchLimit: fetchLimit
  64. )
  65. return try await context.perform {
  66. guard let fetchedResults = results as? [OverrideStored] else {
  67. throw CoreDataError.fetchError(function: #function, file: #file)
  68. }
  69. return fetchedResults.map(\.objectID)
  70. }
  71. }
  72. /// Returns the NSManagedObjectID of the Override Presets
  73. func fetchForOverridePresets() async throws -> [NSManagedObjectID] {
  74. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  75. ofType: OverrideStored.self,
  76. onContext: context,
  77. predicate: NSPredicate.allOverridePresets,
  78. key: "orderPosition",
  79. ascending: true
  80. )
  81. return try await context.perform {
  82. guard let fetchedResults = results as? [OverrideStored] else {
  83. throw CoreDataError.fetchError(function: #function, file: #file)
  84. }
  85. return fetchedResults.map(\.objectID)
  86. }
  87. }
  88. @MainActor func calculateTarget(override: OverrideStored) -> Decimal {
  89. guard let overrideTarget = override.target, overrideTarget != 0 else {
  90. return 0
  91. }
  92. return overrideTarget.decimalValue
  93. }
  94. func storeOverride(override: Override) async throws {
  95. var presetCount = -1
  96. if override.isPreset {
  97. let presets = try await fetchForOverridePresets()
  98. presetCount = presets.count
  99. }
  100. try await context.perform {
  101. let newOverride = OverrideStored(context: self.context)
  102. // override key meta data
  103. if !override.name.isEmpty {
  104. newOverride.name = override.name
  105. } else {
  106. let formattedDate = self.dateFormatter.string(from: Date())
  107. newOverride.name = "Override \(formattedDate)"
  108. }
  109. newOverride.id = UUID().uuidString
  110. newOverride.date = override.date
  111. newOverride.isPreset = override.isPreset
  112. newOverride.isUploadedToNS = false
  113. // Assign orderPosition if it's a preset and presetCount is valid
  114. if override.isPreset, presetCount > -1 {
  115. newOverride.orderPosition = Int16(presetCount + 1) // Ensure type matches Core Data model
  116. }
  117. // override metrics
  118. newOverride.duration = override.duration as NSDecimalNumber
  119. newOverride.indefinite = override.indefinite
  120. newOverride.percentage = override.percentage
  121. newOverride.isfAndCr = override.isfAndCr
  122. newOverride.isf = override.isf
  123. newOverride.cr = override.cr
  124. newOverride.enabled = override.enabled
  125. newOverride.smbIsOff = override.smbIsOff
  126. if override.overrideTarget {
  127. newOverride.target = override.target as NSDecimalNumber
  128. } else {
  129. newOverride.target = 0
  130. }
  131. if override.advancedSettings {
  132. newOverride.advancedSettings = true
  133. newOverride.smbMinutes = override.smbMinutes as NSDecimalNumber
  134. newOverride.uamMinutes = override.uamMinutes as NSDecimalNumber
  135. }
  136. if override.smbIsScheduledOff {
  137. newOverride.smbIsScheduledOff = true
  138. newOverride.start = override.start as NSDecimalNumber
  139. newOverride.end = override.end as NSDecimalNumber
  140. } else {
  141. newOverride.smbIsScheduledOff = false
  142. }
  143. guard self.context.hasChanges else { return }
  144. try self.context.save()
  145. }
  146. }
  147. // Copy the current Override if it is a RUNNING Preset
  148. /// otherwise we would edit the Preset
  149. @MainActor func copyRunningOverride(_ override: OverrideStored) async -> NSManagedObjectID {
  150. let newOverride = OverrideStored(context: viewContext)
  151. newOverride.duration = override.duration
  152. newOverride.indefinite = override.indefinite
  153. newOverride.percentage = override.percentage
  154. newOverride.smbIsOff = override.smbIsOff
  155. newOverride.name = override.name
  156. newOverride.isPreset = false // no Preset
  157. newOverride.date = override.date
  158. newOverride.enabled = override.enabled
  159. newOverride.target = override.target
  160. newOverride.advancedSettings = override.advancedSettings
  161. newOverride.isfAndCr = override.isfAndCr
  162. newOverride.isf = override.isf
  163. newOverride.cr = override.cr
  164. newOverride.smbIsScheduledOff = override.smbIsScheduledOff
  165. newOverride.start = override.start
  166. newOverride.end = override.end
  167. newOverride.smbMinutes = override.smbMinutes
  168. newOverride.uamMinutes = override.uamMinutes
  169. newOverride.isUploadedToNS = true // set to true to avoid getting duplicate entries on NS
  170. await viewContext.perform {
  171. do {
  172. guard self.viewContext.hasChanges else { return }
  173. try self.viewContext.save()
  174. } catch let error as NSError {
  175. debugPrint(
  176. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to copy Override with error: \(error.userInfo)"
  177. )
  178. }
  179. }
  180. return newOverride.objectID
  181. }
  182. /// - Parameter: NSManagedObjectID to be able to transfer the object safely from one thread to another thread
  183. func deleteOverridePreset(_ objectID: NSManagedObjectID) async {
  184. // Use injected context if available, otherwise create new task context
  185. let taskContext = context != CoreDataStack.shared.newTaskContext()
  186. ? context
  187. : CoreDataStack.shared.newTaskContext()
  188. taskContext.name = "deleteContext"
  189. taskContext.transactionAuthor = "deleteOverride"
  190. await taskContext.perform {
  191. do {
  192. guard let override = try taskContext.existingObject(with: objectID) as? OverrideStored else {
  193. debugPrint("Override for batch delete not found. \(DebuggingIdentifiers.failed)")
  194. return
  195. }
  196. taskContext.delete(override)
  197. guard taskContext.hasChanges else { return }
  198. try taskContext.save()
  199. debugPrint(
  200. "OverrideStorage: \(#function) \(DebuggingIdentifiers.succeeded) deleted override from core data"
  201. )
  202. } catch {
  203. debugPrint("\(DebuggingIdentifiers.failed) Error deleting override: \(error.localizedDescription)")
  204. }
  205. }
  206. }
  207. func getOverridesNotYetUploadedToNightscout() async throws -> [NightscoutExercise] {
  208. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  209. ofType: OverrideStored.self,
  210. onContext: context,
  211. predicate: NSPredicate.lastActiveAdjustmentNotYetUploadedToNightscout,
  212. key: "date",
  213. ascending: false
  214. )
  215. return try await context.perform {
  216. guard let fetchedOverrides = results as? [OverrideStored] else {
  217. throw CoreDataError.fetchError(function: #function, file: #file)
  218. }
  219. return fetchedOverrides.map { override in
  220. let duration = override.indefinite ? 43200 : override.duration ?? 0 // 43200 min = 30 days
  221. return NightscoutExercise(
  222. duration: Int(truncating: duration),
  223. eventType: OverrideStored.EventType.nsExercise,
  224. createdAt: override.date ?? Date(),
  225. enteredBy: NightscoutExercise.local,
  226. notes: override.name ?? String(localized: "Custom Override"),
  227. id: UUID(uuidString: override.id ?? UUID().uuidString)
  228. )
  229. }
  230. }
  231. }
  232. func getOverrideRunsNotYetUploadedToNightscout() async throws -> [NightscoutExercise] {
  233. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  234. ofType: OverrideRunStored.self,
  235. onContext: context,
  236. predicate: NSPredicate(
  237. format: "startDate >= %@ AND isUploadedToNS == %@",
  238. Date.oneDayAgo as NSDate,
  239. false as NSNumber
  240. ),
  241. key: "startDate",
  242. ascending: false
  243. )
  244. return try await context.perform {
  245. guard let fetchedOverrideRuns = results as? [OverrideRunStored] else {
  246. throw CoreDataError.fetchError(function: #function, file: #file)
  247. }
  248. return fetchedOverrideRuns.map { overrideRun in
  249. var durationInMinutes = (overrideRun.endDate?.timeIntervalSince(overrideRun.startDate ?? Date()) ?? 1) / 60
  250. durationInMinutes = durationInMinutes < 1 ? 1 : durationInMinutes
  251. return NightscoutExercise(
  252. duration: Int(durationInMinutes),
  253. eventType: OverrideStored.EventType.nsExercise,
  254. createdAt: (overrideRun.startDate ?? overrideRun.override?.date) ?? Date(),
  255. enteredBy: NightscoutExercise.local,
  256. notes: overrideRun.name ?? String(localized: "Custom Override"),
  257. id: overrideRun.id
  258. )
  259. }
  260. }
  261. }
  262. /// This check is needed to force re-rendering of overrides in the Nightscout main chart
  263. /// if the override duration has changed (cancelled, customized or replaced with other override),
  264. /// since just updating durations in existing entries doesn't trigger re-rendering.
  265. func checkIfShouldDeleteNightscoutOverrideEntry(
  266. forCreatedAt createdAtString: String,
  267. newDuration: Int?,
  268. using nightscout: NightscoutAPI
  269. ) async throws {
  270. let formatter = ISO8601DateFormatter()
  271. formatter.formatOptions = [.withInternetDateTime, .withFractionalSeconds]
  272. guard let jsonDate = formatter.date(from: createdAtString) else {
  273. debug(.nightscout, "Could not parse override created_at string: \(createdAtString)")
  274. return
  275. }
  276. /// Define a tolerance window (in seconds)
  277. /// This is neccessary to handle small rounding/conversion time differences
  278. /// when comparing dates between core data and NightscoutExercise json
  279. let tolerance: TimeInterval = 0.1
  280. let lowerBound = jsonDate.addingTimeInterval(-tolerance)
  281. let upperBound = jsonDate.addingTimeInterval(tolerance)
  282. /// Build a predicate to fetch a stored override (from OverrideStored) whose date is within the tolerance window.
  283. let predicate = NSPredicate(format: "date >= %@ AND date <= %@", lowerBound as NSDate, upperBound as NSDate)
  284. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  285. ofType: OverrideStored.self,
  286. onContext: context,
  287. predicate: predicate,
  288. key: "date",
  289. ascending: false
  290. )
  291. let storedOverride: NightscoutExercise? = await context.perform {
  292. guard let fetched = results as? [OverrideStored],
  293. let record = fetched.first,
  294. let recordDate = record.date else { return nil }
  295. let duration = record.indefinite ? 43200 : record.duration ?? 0
  296. return NightscoutExercise(
  297. duration: Int(truncating: duration),
  298. eventType: OverrideStored.EventType.nsExercise,
  299. createdAt: recordDate,
  300. enteredBy: NightscoutExercise.local,
  301. notes: record.name ?? String(localized: "Custom Override"),
  302. id: UUID(uuidString: record.id ?? UUID().uuidString)
  303. )
  304. }
  305. if let existing = storedOverride {
  306. // Only delete existing nightscout entries if the durations differ.
  307. if let existingDuration = existing.duration, let newDuration = newDuration, existingDuration != newDuration {
  308. try await nightscout.deleteNightscoutOverride(withCreatedAt: createdAtString)
  309. }
  310. }
  311. }
  312. func getPresetOverridesForNightscout() async throws -> [NightscoutPresetOverride] {
  313. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  314. ofType: OverrideStored.self,
  315. onContext: context,
  316. predicate: NSPredicate.allOverridePresets,
  317. key: "orderPosition",
  318. ascending: true
  319. )
  320. return try await context.perform {
  321. guard let fetchedResults = results as? [OverrideStored] else {
  322. throw CoreDataError.fetchError(function: #function, file: #file)
  323. }
  324. return fetchedResults.map { overrideStored in
  325. let duration = overrideStored.duration as? Decimal != 0 ? overrideStored.duration as? Decimal : nil
  326. let percentage = overrideStored.percentage != 0 ? overrideStored.percentage : nil
  327. let target = (overrideStored.target as? Decimal) != 0 ? overrideStored.target as? Decimal : nil
  328. return NightscoutPresetOverride(
  329. name: overrideStored.name ?? "",
  330. duration: duration,
  331. percentage: percentage,
  332. target: target
  333. )
  334. }
  335. }
  336. }
  337. func fetchLatestActiveOverride() async throws -> NSManagedObjectID? {
  338. let results = try await CoreDataStack.shared.fetchEntitiesAsync(
  339. ofType: OverrideStored.self,
  340. onContext: context,
  341. predicate: NSPredicate.lastActiveOverride,
  342. key: "date",
  343. ascending: false,
  344. fetchLimit: 1
  345. )
  346. return try await context.perform {
  347. guard let fetchedResults = results as? [OverrideStored]
  348. else {
  349. throw CoreDataError.fetchError(function: #function, file: #file)
  350. }
  351. return fetchedResults.first?.objectID
  352. }
  353. }
  354. }