PumpHistoryStorage.swift 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  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. fat: nil,
  219. protein: nil,
  220. targetTop: nil,
  221. targetBottom: nil
  222. ))
  223. case .tempBasalDuration:
  224. if var last = result.popLast(), last.eventType == .nsTempBasal, last.createdAt == event.timestamp {
  225. last.duration = event.durationMin
  226. last.rawDuration = event
  227. result.append(last)
  228. }
  229. default: break
  230. }
  231. return result
  232. }
  233. let bolusesAndCarbs = events.compactMap { event -> NigtscoutTreatment? in
  234. switch event.type {
  235. case .bolus:
  236. return NigtscoutTreatment(
  237. duration: event.duration,
  238. rawDuration: nil,
  239. rawRate: nil,
  240. absolute: nil,
  241. rate: nil,
  242. eventType: .bolus,
  243. createdAt: event.timestamp,
  244. enteredBy: NigtscoutTreatment.local,
  245. bolus: event,
  246. insulin: event.amount,
  247. notes: nil,
  248. carbs: nil,
  249. fat: nil,
  250. protein: nil,
  251. targetTop: nil,
  252. targetBottom: nil
  253. )
  254. case .journalCarbs:
  255. return NigtscoutTreatment(
  256. duration: nil,
  257. rawDuration: nil,
  258. rawRate: nil,
  259. absolute: nil,
  260. rate: nil,
  261. eventType: .nsCarbCorrection,
  262. createdAt: event.timestamp,
  263. enteredBy: NigtscoutTreatment.local,
  264. bolus: nil,
  265. insulin: nil,
  266. notes: nil,
  267. carbs: Decimal(event.carbInput ?? 0),
  268. fat: nil,
  269. protein: nil,
  270. targetTop: nil,
  271. targetBottom: nil
  272. )
  273. default: return nil
  274. }
  275. }
  276. let misc = events.compactMap { event -> NigtscoutTreatment? in
  277. switch event.type {
  278. case .prime:
  279. return NigtscoutTreatment(
  280. duration: event.duration,
  281. rawDuration: nil,
  282. rawRate: nil,
  283. absolute: nil,
  284. rate: nil,
  285. eventType: .nsSiteChange,
  286. createdAt: event.timestamp,
  287. enteredBy: NigtscoutTreatment.local,
  288. bolus: event,
  289. insulin: nil,
  290. notes: nil,
  291. carbs: nil,
  292. fat: nil,
  293. protein: nil,
  294. targetTop: nil,
  295. targetBottom: nil
  296. )
  297. case .rewind:
  298. return NigtscoutTreatment(
  299. duration: nil,
  300. rawDuration: nil,
  301. rawRate: nil,
  302. absolute: nil,
  303. rate: nil,
  304. eventType: .nsInsulinChange,
  305. createdAt: event.timestamp,
  306. enteredBy: NigtscoutTreatment.local,
  307. bolus: nil,
  308. insulin: nil,
  309. notes: nil,
  310. carbs: nil,
  311. fat: nil,
  312. protein: nil,
  313. targetTop: nil,
  314. targetBottom: nil
  315. )
  316. case .pumpAlarm:
  317. return NigtscoutTreatment(
  318. duration: 30, // minutes
  319. rawDuration: nil,
  320. rawRate: nil,
  321. absolute: nil,
  322. rate: nil,
  323. eventType: .nsAnnouncement,
  324. createdAt: event.timestamp,
  325. enteredBy: NigtscoutTreatment.local,
  326. bolus: nil,
  327. insulin: nil,
  328. notes: "Alarm \(String(describing: event.note)) \(event.type)",
  329. carbs: nil,
  330. fat: nil,
  331. protein: nil,
  332. targetTop: nil,
  333. targetBottom: nil
  334. )
  335. default: return nil
  336. }
  337. }
  338. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedPumphistory, as: [NigtscoutTreatment].self) ?? []
  339. let treatments = Array(Set([bolusesAndCarbs, temps, misc].flatMap { $0 }).subtracting(Set(uploaded)))
  340. return treatments.sorted { $0.createdAt! > $1.createdAt! }
  341. }
  342. func saveCancelTempEvents() {
  343. let basalID = UUID().uuidString
  344. let date = Date()
  345. let events = [
  346. PumpHistoryEvent(
  347. id: basalID,
  348. type: .tempBasalDuration,
  349. timestamp: date,
  350. amount: nil,
  351. duration: nil,
  352. durationMin: 0,
  353. rate: nil,
  354. temp: nil,
  355. carbInput: nil
  356. ),
  357. PumpHistoryEvent(
  358. id: "_" + basalID,
  359. type: .tempBasal,
  360. timestamp: date,
  361. amount: nil,
  362. duration: nil,
  363. durationMin: nil,
  364. rate: 0,
  365. temp: .absolute,
  366. carbInput: nil
  367. )
  368. ]
  369. storeEvents(events)
  370. }
  371. }