PumpHistoryStorage.swift 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. import Foundation
  2. import LoopKit
  3. import SwiftDate
  4. import Swinject
  5. protocol PumpHistoryObserver {
  6. func pumpHistoryDidUpdate(_ events: [PumpHistoryEvent])
  7. }
  8. protocol PumpHistoryStorage {
  9. func storePumpEvents(_ events: [NewPumpEvent])
  10. func storeEvents(_ events: [PumpHistoryEvent])
  11. func storeJournalCarbs(_ carbs: Int)
  12. func recent() -> [PumpHistoryEvent]
  13. func nightscoutTretmentsNotUploaded() -> [NigtscoutTreatment]
  14. func saveCancelTempEvents()
  15. func deleteInsulin(at date: Date)
  16. }
  17. final class BasePumpHistoryStorage: PumpHistoryStorage, Injectable {
  18. private let processQueue = DispatchQueue(label: "BasePumpHistoryStorage.processQueue")
  19. @Injected() private var storage: FileStorage!
  20. @Injected() private var broadcaster: Broadcaster!
  21. init(resolver: Resolver) {
  22. injectServices(resolver)
  23. }
  24. private let context = CoreDataStack.shared.backgroundContext
  25. func storePumpEvents(_ events: [NewPumpEvent]) {
  26. processQueue.async {
  27. let eventsToStore = events.flatMap { event -> [PumpHistoryEvent] in
  28. let id = event.raw.md5String
  29. switch event.type {
  30. case .bolus:
  31. guard let dose = event.dose else { return [] }
  32. let amount = Decimal(string: dose.unitsInDeliverableIncrements.description)
  33. let minutes = Int((dose.endDate - dose.startDate).timeInterval / 60)
  34. // MARK: - save to Core Data
  35. self.context.perform {
  36. let new = InsulinStored(context: self.context)
  37. new.amount = amount as? NSDecimalNumber
  38. new.date = Date()
  39. new.external = false
  40. new.id = UUID()
  41. new.isSMB = true
  42. if self.context.hasChanges {
  43. do {
  44. try self.context.save()
  45. debugPrint(
  46. "Pump History storage: \(#function) \(CoreDataStack.identifier) \(DebuggingIdentifiers.succeeded) saved smbs to core data"
  47. )
  48. } catch {
  49. debugPrint(
  50. "Pump History storage: \(#function) \(CoreDataStack.identifier) \(DebuggingIdentifiers.failed) failed to save smbs to core data"
  51. )
  52. }
  53. }
  54. }
  55. return [PumpHistoryEvent(
  56. id: id,
  57. type: .bolus,
  58. timestamp: event.date,
  59. amount: amount,
  60. duration: minutes,
  61. durationMin: nil,
  62. rate: nil,
  63. temp: nil,
  64. carbInput: nil,
  65. isSMB: dose.automatic,
  66. isExternal: dose.manuallyEntered
  67. )]
  68. case .tempBasal:
  69. guard let dose = event.dose else { return [] }
  70. let rate = Decimal(dose.unitsPerHour)
  71. let minutes = (dose.endDate - dose.startDate).timeInterval / 60
  72. let delivered = dose.deliveredUnits
  73. let date = event.date
  74. let isCancel = delivered != nil //! event.isMutable && delivered != nil
  75. guard !isCancel else { return [] }
  76. return [
  77. PumpHistoryEvent(
  78. id: id,
  79. type: .tempBasalDuration,
  80. timestamp: date,
  81. amount: nil,
  82. duration: nil,
  83. durationMin: Int(round(minutes)),
  84. rate: nil,
  85. temp: nil,
  86. carbInput: nil
  87. ),
  88. PumpHistoryEvent(
  89. id: "_" + id,
  90. type: .tempBasal,
  91. timestamp: date,
  92. amount: nil,
  93. duration: nil,
  94. durationMin: nil,
  95. rate: rate,
  96. temp: .absolute,
  97. carbInput: nil
  98. )
  99. ]
  100. case .suspend:
  101. return [
  102. PumpHistoryEvent(
  103. id: id,
  104. type: .pumpSuspend,
  105. timestamp: event.date,
  106. amount: nil,
  107. duration: nil,
  108. durationMin: nil,
  109. rate: nil,
  110. temp: nil,
  111. carbInput: nil
  112. )
  113. ]
  114. case .resume:
  115. return [
  116. PumpHistoryEvent(
  117. id: id,
  118. type: .pumpResume,
  119. timestamp: event.date,
  120. amount: nil,
  121. duration: nil,
  122. durationMin: nil,
  123. rate: nil,
  124. temp: nil,
  125. carbInput: nil
  126. )
  127. ]
  128. case .rewind:
  129. return [
  130. PumpHistoryEvent(
  131. id: id,
  132. type: .rewind,
  133. timestamp: event.date,
  134. amount: nil,
  135. duration: nil,
  136. durationMin: nil,
  137. rate: nil,
  138. temp: nil,
  139. carbInput: nil
  140. )
  141. ]
  142. case .prime:
  143. return [
  144. PumpHistoryEvent(
  145. id: id,
  146. type: .prime,
  147. timestamp: event.date,
  148. amount: nil,
  149. duration: nil,
  150. durationMin: nil,
  151. rate: nil,
  152. temp: nil,
  153. carbInput: nil
  154. )
  155. ]
  156. case .alarm:
  157. return [
  158. PumpHistoryEvent(
  159. id: id,
  160. type: .pumpAlarm,
  161. timestamp: event.date,
  162. note: event.title
  163. )
  164. ]
  165. default:
  166. return []
  167. }
  168. }
  169. self.storeEvents(eventsToStore)
  170. }
  171. }
  172. func storeJournalCarbs(_ carbs: Int) {
  173. processQueue.async {
  174. let eventsToStore = [
  175. PumpHistoryEvent(
  176. id: UUID().uuidString,
  177. type: .journalCarbs,
  178. timestamp: Date(),
  179. amount: nil,
  180. duration: nil,
  181. durationMin: nil,
  182. rate: nil,
  183. temp: nil,
  184. carbInput: carbs
  185. )
  186. ]
  187. self.storeEvents(eventsToStore)
  188. }
  189. }
  190. func storeEvents(_ events: [PumpHistoryEvent]) {
  191. processQueue.async {
  192. let file = OpenAPS.Monitor.pumpHistory
  193. var uniqEvents: [PumpHistoryEvent] = []
  194. self.storage.transaction { storage in
  195. storage.append(events, to: file, uniqBy: \.id)
  196. uniqEvents = storage.retrieve(file, as: [PumpHistoryEvent].self)?
  197. .filter { $0.timestamp.addingTimeInterval(1.days.timeInterval) > Date() }
  198. .sorted { $0.timestamp > $1.timestamp } ?? []
  199. storage.save(Array(uniqEvents), as: file)
  200. }
  201. self.broadcaster.notify(PumpHistoryObserver.self, on: self.processQueue) {
  202. $0.pumpHistoryDidUpdate(uniqEvents)
  203. }
  204. }
  205. }
  206. func recent() -> [PumpHistoryEvent] {
  207. storage.retrieve(OpenAPS.Monitor.pumpHistory, as: [PumpHistoryEvent].self)?.reversed() ?? []
  208. }
  209. func deleteInsulin(at date: Date) {
  210. processQueue.sync {
  211. var allValues = storage.retrieve(OpenAPS.Monitor.pumpHistory, as: [PumpHistoryEvent].self) ?? []
  212. guard let entryIndex = allValues.firstIndex(where: { $0.timestamp == date }) else {
  213. return
  214. }
  215. allValues.remove(at: entryIndex)
  216. storage.save(allValues, as: OpenAPS.Monitor.pumpHistory)
  217. broadcaster.notify(PumpHistoryObserver.self, on: processQueue) {
  218. $0.pumpHistoryDidUpdate(allValues)
  219. }
  220. }
  221. }
  222. func determineBolusEventType(for event: PumpHistoryEvent) -> EventType {
  223. if event.isSMB ?? false {
  224. return .smb
  225. }
  226. if event.isExternal ?? false {
  227. return .isExternal
  228. }
  229. return event.type
  230. }
  231. func nightscoutTretmentsNotUploaded() -> [NigtscoutTreatment] {
  232. let events = recent()
  233. guard !events.isEmpty else { return [] }
  234. let temps: [NigtscoutTreatment] = events.reduce([]) { result, event in
  235. var result = result
  236. switch event.type {
  237. case .tempBasal:
  238. result.append(NigtscoutTreatment(
  239. duration: nil,
  240. rawDuration: nil,
  241. rawRate: event,
  242. absolute: event.rate,
  243. rate: event.rate,
  244. eventType: .nsTempBasal,
  245. createdAt: event.timestamp,
  246. enteredBy: NigtscoutTreatment.local,
  247. bolus: nil,
  248. insulin: nil,
  249. notes: nil,
  250. carbs: nil,
  251. fat: nil,
  252. protein: nil,
  253. targetTop: nil,
  254. targetBottom: nil
  255. ))
  256. case .tempBasalDuration:
  257. if var last = result.popLast(), last.eventType == .nsTempBasal, last.createdAt == event.timestamp {
  258. last.duration = event.durationMin
  259. last.rawDuration = event
  260. result.append(last)
  261. }
  262. default: break
  263. }
  264. return result
  265. }
  266. let bolusesAndCarbs = events.compactMap { event -> NigtscoutTreatment? in
  267. switch event.type {
  268. case .bolus:
  269. let eventType = determineBolusEventType(for: event)
  270. return NigtscoutTreatment(
  271. duration: event.duration,
  272. rawDuration: nil,
  273. rawRate: nil,
  274. absolute: nil,
  275. rate: nil,
  276. eventType: eventType,
  277. createdAt: event.timestamp,
  278. enteredBy: NigtscoutTreatment.local,
  279. bolus: event,
  280. insulin: event.amount,
  281. notes: nil,
  282. carbs: nil,
  283. fat: nil,
  284. protein: nil,
  285. targetTop: nil,
  286. targetBottom: nil
  287. )
  288. case .journalCarbs:
  289. return NigtscoutTreatment(
  290. duration: nil,
  291. rawDuration: nil,
  292. rawRate: nil,
  293. absolute: nil,
  294. rate: nil,
  295. eventType: .nsCarbCorrection,
  296. createdAt: event.timestamp,
  297. enteredBy: NigtscoutTreatment.local,
  298. bolus: nil,
  299. insulin: nil,
  300. notes: nil,
  301. carbs: Decimal(event.carbInput ?? 0),
  302. fat: nil,
  303. protein: nil,
  304. targetTop: nil,
  305. targetBottom: nil
  306. )
  307. default: return nil
  308. }
  309. }
  310. let misc = events.compactMap { event -> NigtscoutTreatment? in
  311. switch event.type {
  312. case .prime:
  313. return NigtscoutTreatment(
  314. duration: event.duration,
  315. rawDuration: nil,
  316. rawRate: nil,
  317. absolute: nil,
  318. rate: nil,
  319. eventType: .nsSiteChange,
  320. createdAt: event.timestamp,
  321. enteredBy: NigtscoutTreatment.local,
  322. bolus: event,
  323. insulin: nil,
  324. notes: nil,
  325. carbs: nil,
  326. fat: nil,
  327. protein: nil,
  328. targetTop: nil,
  329. targetBottom: nil
  330. )
  331. case .rewind:
  332. return NigtscoutTreatment(
  333. duration: nil,
  334. rawDuration: nil,
  335. rawRate: nil,
  336. absolute: nil,
  337. rate: nil,
  338. eventType: .nsInsulinChange,
  339. createdAt: event.timestamp,
  340. enteredBy: NigtscoutTreatment.local,
  341. bolus: nil,
  342. insulin: nil,
  343. notes: nil,
  344. carbs: nil,
  345. fat: nil,
  346. protein: nil,
  347. targetTop: nil,
  348. targetBottom: nil
  349. )
  350. case .pumpAlarm:
  351. return NigtscoutTreatment(
  352. duration: 30, // minutes
  353. rawDuration: nil,
  354. rawRate: nil,
  355. absolute: nil,
  356. rate: nil,
  357. eventType: .nsAnnouncement,
  358. createdAt: event.timestamp,
  359. enteredBy: NigtscoutTreatment.local,
  360. bolus: nil,
  361. insulin: nil,
  362. notes: "Alarm \(String(describing: event.note)) \(event.type)",
  363. carbs: nil,
  364. fat: nil,
  365. protein: nil,
  366. targetTop: nil,
  367. targetBottom: nil
  368. )
  369. default: return nil
  370. }
  371. }
  372. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedPumphistory, as: [NigtscoutTreatment].self) ?? []
  373. let treatments = Array(Set([bolusesAndCarbs, temps, misc].flatMap { $0 }).subtracting(Set(uploaded)))
  374. return treatments.sorted { $0.createdAt! > $1.createdAt! }
  375. }
  376. func saveCancelTempEvents() {
  377. let basalID = UUID().uuidString
  378. let date = Date()
  379. let events = [
  380. PumpHistoryEvent(
  381. id: basalID,
  382. type: .tempBasalDuration,
  383. timestamp: date,
  384. amount: nil,
  385. duration: nil,
  386. durationMin: 0,
  387. rate: nil,
  388. temp: nil,
  389. carbInput: nil
  390. ),
  391. PumpHistoryEvent(
  392. id: "_" + basalID,
  393. type: .tempBasal,
  394. timestamp: date,
  395. amount: nil,
  396. duration: nil,
  397. durationMin: nil,
  398. rate: 0,
  399. temp: .absolute,
  400. carbInput: nil
  401. )
  402. ]
  403. storeEvents(events)
  404. }
  405. }