PumpHistoryStorage.swift 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  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. func storePumpEvents(_ events: [NewPumpEvent]) {
  25. processQueue.async {
  26. let eventsToStore = events.flatMap { event -> [PumpHistoryEvent] in
  27. let id = event.raw.md5String
  28. switch event.type {
  29. case .bolus:
  30. guard let dose = event.dose else { return [] }
  31. let amount = Decimal(string: dose.unitsInDeliverableIncrements.description)
  32. let minutes = Int((dose.endDate - dose.startDate).timeInterval / 60)
  33. return [PumpHistoryEvent(
  34. id: id,
  35. type: .bolus,
  36. timestamp: event.date,
  37. amount: amount,
  38. duration: minutes,
  39. durationMin: nil,
  40. rate: nil,
  41. temp: nil,
  42. carbInput: nil
  43. )]
  44. case .tempBasal:
  45. guard let dose = event.dose else { return [] }
  46. let rate = Decimal(dose.unitsPerHour)
  47. let minutes = (dose.endDate - dose.startDate).timeInterval / 60
  48. let delivered = dose.deliveredUnits
  49. let date = event.date
  50. let isCancel = delivered != nil //! event.isMutable && delivered != nil
  51. guard !isCancel else { return [] }
  52. return [
  53. PumpHistoryEvent(
  54. id: id,
  55. type: .tempBasalDuration,
  56. timestamp: date,
  57. amount: nil,
  58. duration: nil,
  59. durationMin: Int(round(minutes)),
  60. rate: nil,
  61. temp: nil,
  62. carbInput: nil
  63. ),
  64. PumpHistoryEvent(
  65. id: "_" + id,
  66. type: .tempBasal,
  67. timestamp: date,
  68. amount: nil,
  69. duration: nil,
  70. durationMin: nil,
  71. rate: rate,
  72. temp: .absolute,
  73. carbInput: nil
  74. )
  75. ]
  76. case .suspend:
  77. return [
  78. PumpHistoryEvent(
  79. id: id,
  80. type: .pumpSuspend,
  81. timestamp: event.date,
  82. amount: nil,
  83. duration: nil,
  84. durationMin: nil,
  85. rate: nil,
  86. temp: nil,
  87. carbInput: nil
  88. )
  89. ]
  90. case .resume:
  91. return [
  92. PumpHistoryEvent(
  93. id: id,
  94. type: .pumpResume,
  95. timestamp: event.date,
  96. amount: nil,
  97. duration: nil,
  98. durationMin: nil,
  99. rate: nil,
  100. temp: nil,
  101. carbInput: nil
  102. )
  103. ]
  104. case .rewind:
  105. return [
  106. PumpHistoryEvent(
  107. id: id,
  108. type: .rewind,
  109. timestamp: event.date,
  110. amount: nil,
  111. duration: nil,
  112. durationMin: nil,
  113. rate: nil,
  114. temp: nil,
  115. carbInput: nil
  116. )
  117. ]
  118. case .prime:
  119. return [
  120. PumpHistoryEvent(
  121. id: id,
  122. type: .prime,
  123. timestamp: event.date,
  124. amount: nil,
  125. duration: nil,
  126. durationMin: nil,
  127. rate: nil,
  128. temp: nil,
  129. carbInput: nil
  130. )
  131. ]
  132. case .alarm:
  133. return [
  134. PumpHistoryEvent(
  135. id: id,
  136. type: .pumpAlarm,
  137. timestamp: event.date,
  138. note: event.title
  139. )
  140. ]
  141. default:
  142. return []
  143. }
  144. }
  145. self.storeEvents(eventsToStore)
  146. }
  147. }
  148. func storeJournalCarbs(_ carbs: Int) {
  149. processQueue.async {
  150. let eventsToStore = [
  151. PumpHistoryEvent(
  152. id: UUID().uuidString,
  153. type: .journalCarbs,
  154. timestamp: Date(),
  155. amount: nil,
  156. duration: nil,
  157. durationMin: nil,
  158. rate: nil,
  159. temp: nil,
  160. carbInput: carbs
  161. )
  162. ]
  163. self.storeEvents(eventsToStore)
  164. }
  165. }
  166. func storeEvents(_ events: [PumpHistoryEvent]) {
  167. processQueue.async {
  168. let file = OpenAPS.Monitor.pumpHistory
  169. var uniqEvents: [PumpHistoryEvent] = []
  170. self.storage.transaction { storage in
  171. storage.append(events, to: file, uniqBy: \.id)
  172. uniqEvents = storage.retrieve(file, as: [PumpHistoryEvent].self)?
  173. .filter { $0.timestamp.addingTimeInterval(1.days.timeInterval) > Date() }
  174. .sorted { $0.timestamp > $1.timestamp } ?? []
  175. storage.save(Array(uniqEvents), as: file)
  176. }
  177. self.broadcaster.notify(PumpHistoryObserver.self, on: self.processQueue) {
  178. $0.pumpHistoryDidUpdate(uniqEvents)
  179. }
  180. }
  181. }
  182. func recent() -> [PumpHistoryEvent] {
  183. storage.retrieve(OpenAPS.Monitor.pumpHistory, as: [PumpHistoryEvent].self)?.reversed() ?? []
  184. }
  185. func deleteInsulin(at date: Date) {
  186. processQueue.sync {
  187. var allValues = storage.retrieve(OpenAPS.Monitor.pumpHistory, as: [PumpHistoryEvent].self) ?? []
  188. guard let entryIndex = allValues.firstIndex(where: { $0.timestamp == date }) else {
  189. return
  190. }
  191. allValues.remove(at: entryIndex)
  192. storage.save(allValues, as: OpenAPS.Monitor.pumpHistory)
  193. broadcaster.notify(PumpHistoryObserver.self, on: processQueue) {
  194. $0.pumpHistoryDidUpdate(allValues)
  195. }
  196. }
  197. }
  198. func nightscoutTretmentsNotUploaded() -> [NigtscoutTreatment] {
  199. let events = recent()
  200. guard !events.isEmpty else { return [] }
  201. let temps: [NigtscoutTreatment] = events.reduce([]) { result, event in
  202. var result = result
  203. switch event.type {
  204. case .tempBasal:
  205. result.append(NigtscoutTreatment(
  206. duration: nil,
  207. rawDuration: nil,
  208. rawRate: event,
  209. absolute: event.rate,
  210. rate: event.rate,
  211. eventType: .nsTempBasal,
  212. createdAt: event.timestamp,
  213. enteredBy: NigtscoutTreatment.local,
  214. bolus: nil,
  215. insulin: nil,
  216. notes: nil,
  217. carbs: nil,
  218. targetTop: nil,
  219. targetBottom: nil
  220. ))
  221. case .tempBasalDuration:
  222. if var last = result.popLast(), last.eventType == .nsTempBasal, last.createdAt == event.timestamp {
  223. last.duration = event.durationMin
  224. last.rawDuration = event
  225. result.append(last)
  226. }
  227. default: break
  228. }
  229. return result
  230. }
  231. let bolusesAndCarbs = events.compactMap { event -> NigtscoutTreatment? in
  232. switch event.type {
  233. case .bolus:
  234. return NigtscoutTreatment(
  235. duration: event.duration,
  236. rawDuration: nil,
  237. rawRate: nil,
  238. absolute: nil,
  239. rate: nil,
  240. eventType: .bolus,
  241. createdAt: event.timestamp,
  242. enteredBy: NigtscoutTreatment.local,
  243. bolus: event,
  244. insulin: event.amount,
  245. notes: nil,
  246. carbs: nil,
  247. targetTop: nil,
  248. targetBottom: nil
  249. )
  250. case .journalCarbs:
  251. return NigtscoutTreatment(
  252. duration: nil,
  253. rawDuration: nil,
  254. rawRate: nil,
  255. absolute: nil,
  256. rate: nil,
  257. eventType: .nsCarbCorrection,
  258. createdAt: event.timestamp,
  259. enteredBy: NigtscoutTreatment.local,
  260. bolus: nil,
  261. insulin: nil,
  262. notes: nil,
  263. carbs: Decimal(event.carbInput ?? 0),
  264. targetTop: nil,
  265. targetBottom: nil
  266. )
  267. default: return nil
  268. }
  269. }
  270. let misc = events.compactMap { event -> NigtscoutTreatment? in
  271. switch event.type {
  272. case .prime:
  273. return NigtscoutTreatment(
  274. duration: event.duration,
  275. rawDuration: nil,
  276. rawRate: nil,
  277. absolute: nil,
  278. rate: nil,
  279. eventType: .nsSiteChange,
  280. createdAt: event.timestamp,
  281. enteredBy: NigtscoutTreatment.local,
  282. bolus: event,
  283. insulin: nil,
  284. notes: nil,
  285. carbs: nil,
  286. targetTop: nil,
  287. targetBottom: nil
  288. )
  289. case .rewind:
  290. return NigtscoutTreatment(
  291. duration: nil,
  292. rawDuration: nil,
  293. rawRate: nil,
  294. absolute: nil,
  295. rate: nil,
  296. eventType: .nsInsulinChange,
  297. createdAt: event.timestamp,
  298. enteredBy: NigtscoutTreatment.local,
  299. bolus: nil,
  300. insulin: nil,
  301. notes: nil,
  302. carbs: nil,
  303. targetTop: nil,
  304. targetBottom: nil
  305. )
  306. case .pumpAlarm:
  307. return NigtscoutTreatment(
  308. duration: 30, // minutes
  309. rawDuration: nil,
  310. rawRate: nil,
  311. absolute: nil,
  312. rate: nil,
  313. eventType: .nsAnnouncement,
  314. createdAt: event.timestamp,
  315. enteredBy: NigtscoutTreatment.local,
  316. bolus: nil,
  317. insulin: nil,
  318. notes: "Alarm \(String(describing: event.note)) \(event.type)",
  319. carbs: nil,
  320. targetTop: nil,
  321. targetBottom: nil
  322. )
  323. default: return nil
  324. }
  325. }
  326. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedPumphistory, as: [NigtscoutTreatment].self) ?? []
  327. let treatments = Array(Set([bolusesAndCarbs, temps, misc].flatMap { $0 }).subtracting(Set(uploaded)))
  328. return treatments.sorted { $0.createdAt! > $1.createdAt! }
  329. }
  330. func saveCancelTempEvents() {
  331. let basalID = UUID().uuidString
  332. let date = Date()
  333. let events = [
  334. PumpHistoryEvent(
  335. id: basalID,
  336. type: .tempBasalDuration,
  337. timestamp: date,
  338. amount: nil,
  339. duration: nil,
  340. durationMin: 0,
  341. rate: nil,
  342. temp: nil,
  343. carbInput: nil
  344. ),
  345. PumpHistoryEvent(
  346. id: "_" + basalID,
  347. type: .tempBasal,
  348. timestamp: date,
  349. amount: nil,
  350. duration: nil,
  351. durationMin: nil,
  352. rate: 0,
  353. temp: .absolute,
  354. carbInput: nil
  355. )
  356. ]
  357. storeEvents(events)
  358. }
  359. }