PumpHistoryStorage.swift 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  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 storeJournalCarbs(_ carbs: Int)
  11. func recent() -> [PumpHistoryEvent]
  12. func nightscoutTretmentsNotUploaded() -> [NigtscoutTreatment]
  13. }
  14. final class BasePumpHistoryStorage: PumpHistoryStorage, Injectable {
  15. private let processQueue = DispatchQueue(label: "BasePumpHistoryStorage.processQueue")
  16. @Injected() private var storage: FileStorage!
  17. @Injected() private var broadcaster: Broadcaster!
  18. init(resolver: Resolver) {
  19. injectServices(resolver)
  20. }
  21. func storePumpEvents(_ events: [NewPumpEvent]) {
  22. processQueue.async {
  23. let eventsToStore = events.flatMap { event -> [PumpHistoryEvent] in
  24. let id = event.raw.md5String
  25. switch event.type {
  26. case .bolus:
  27. guard let dose = event.dose else { return [] }
  28. let amount = Decimal(string: dose.unitsInDeliverableIncrements.description)
  29. let minutes = Int((dose.endDate - dose.startDate).timeInterval / 60)
  30. return [PumpHistoryEvent(
  31. id: id,
  32. type: .bolus,
  33. timestamp: event.date,
  34. amount: amount,
  35. duration: minutes,
  36. durationMin: nil,
  37. rate: nil,
  38. temp: nil,
  39. carbInput: nil
  40. )]
  41. case .tempBasal:
  42. guard let dose = event.dose else { return [] }
  43. let rate = Decimal(string: dose.unitsPerHour.description)
  44. let minutes = Int((dose.endDate - dose.startDate).timeInterval / 60)
  45. return [
  46. PumpHistoryEvent(
  47. id: id,
  48. type: .tempBasalDuration,
  49. timestamp: event.date,
  50. amount: nil,
  51. duration: nil,
  52. durationMin: minutes,
  53. rate: nil,
  54. temp: nil,
  55. carbInput: nil
  56. ),
  57. PumpHistoryEvent(
  58. id: "_" + id,
  59. type: .tempBasal,
  60. timestamp: event.date,
  61. amount: nil,
  62. duration: nil,
  63. durationMin: nil,
  64. rate: rate,
  65. temp: .absolute,
  66. carbInput: nil
  67. )
  68. ]
  69. case .suspend:
  70. return [
  71. PumpHistoryEvent(
  72. id: id,
  73. type: .pumpSuspend,
  74. timestamp: event.date,
  75. amount: nil,
  76. duration: nil,
  77. durationMin: nil,
  78. rate: nil,
  79. temp: nil,
  80. carbInput: nil
  81. )
  82. ]
  83. case .resume:
  84. return [
  85. PumpHistoryEvent(
  86. id: id,
  87. type: .pumpResume,
  88. timestamp: event.date,
  89. amount: nil,
  90. duration: nil,
  91. durationMin: nil,
  92. rate: nil,
  93. temp: nil,
  94. carbInput: nil
  95. )
  96. ]
  97. case .rewind:
  98. return [
  99. PumpHistoryEvent(
  100. id: id,
  101. type: .rewind,
  102. timestamp: event.date,
  103. amount: nil,
  104. duration: nil,
  105. durationMin: nil,
  106. rate: nil,
  107. temp: nil,
  108. carbInput: nil
  109. )
  110. ]
  111. case .prime:
  112. return [
  113. PumpHistoryEvent(
  114. id: id,
  115. type: .prime,
  116. timestamp: event.date,
  117. amount: nil,
  118. duration: nil,
  119. durationMin: nil,
  120. rate: nil,
  121. temp: nil,
  122. carbInput: nil
  123. )
  124. ]
  125. default:
  126. return []
  127. }
  128. }
  129. self.processNewEvents(eventsToStore)
  130. }
  131. }
  132. func storeJournalCarbs(_ carbs: Int) {
  133. processQueue.async {
  134. let eventsToStore = [
  135. PumpHistoryEvent(
  136. id: UUID().uuidString,
  137. type: .journalCarbs,
  138. timestamp: Date(),
  139. amount: nil,
  140. duration: nil,
  141. durationMin: nil,
  142. rate: nil,
  143. temp: nil,
  144. carbInput: carbs
  145. )
  146. ]
  147. self.processNewEvents(eventsToStore)
  148. }
  149. }
  150. private func processNewEvents(_ events: [PumpHistoryEvent]) {
  151. dispatchPrecondition(condition: .onQueue(processQueue))
  152. let file = OpenAPS.Monitor.pumpHistory
  153. var uniqEvents: [PumpHistoryEvent] = []
  154. storage.transaction { storage in
  155. storage.append(events, to: file, uniqBy: \.id)
  156. uniqEvents = storage.retrieve(file, as: [PumpHistoryEvent].self)?
  157. .filter { $0.timestamp.addingTimeInterval(1.days.timeInterval) > Date() }
  158. .sorted { $0.timestamp > $1.timestamp } ?? []
  159. storage.save(Array(uniqEvents), as: file)
  160. }
  161. broadcaster.notify(PumpHistoryObserver.self, on: processQueue) {
  162. $0.pumpHistoryDidUpdate(uniqEvents)
  163. }
  164. }
  165. func recent() -> [PumpHistoryEvent] {
  166. storage.retrieve(OpenAPS.Monitor.pumpHistory, as: [PumpHistoryEvent].self)?.reversed() ?? []
  167. }
  168. func nightscoutTretmentsNotUploaded() -> [NigtscoutTreatment] {
  169. let events = recent()
  170. guard !events.isEmpty else { return [] }
  171. let temps: [NigtscoutTreatment] = events.reduce([]) { result, event in
  172. var result = result
  173. switch event.type {
  174. case .tempBasal:
  175. result.append(NigtscoutTreatment(
  176. duration: nil,
  177. rawDuration: nil,
  178. rawRate: event,
  179. absolute: event.rate,
  180. rate: event.rate,
  181. eventType: .nsTempBasal,
  182. createdAt: event.timestamp,
  183. enteredBy: NigtscoutTreatment.local,
  184. bolus: nil,
  185. insulin: nil,
  186. notes: nil,
  187. carbs: nil,
  188. targetTop: nil,
  189. targetBottom: nil
  190. ))
  191. case .tempBasalDuration:
  192. if var last = result.popLast(), last.eventType == .nsTempBasal, last.createdAt == event.timestamp {
  193. last.duration = event.durationMin
  194. last.rawDuration = event
  195. result.append(last)
  196. }
  197. default: break
  198. }
  199. return result
  200. }
  201. let bolusesAndCarbs = events.compactMap { event -> NigtscoutTreatment? in
  202. switch event.type {
  203. case .bolus:
  204. return NigtscoutTreatment(
  205. duration: event.duration,
  206. rawDuration: nil,
  207. rawRate: nil,
  208. absolute: nil,
  209. rate: nil,
  210. eventType: .bolus,
  211. createdAt: event.timestamp,
  212. enteredBy: NigtscoutTreatment.local,
  213. bolus: event,
  214. insulin: event.amount,
  215. notes: nil,
  216. carbs: nil,
  217. targetTop: nil,
  218. targetBottom: nil
  219. )
  220. case .journalCarbs:
  221. return NigtscoutTreatment(
  222. duration: nil,
  223. rawDuration: nil,
  224. rawRate: nil,
  225. absolute: nil,
  226. rate: nil,
  227. eventType: .nsCarbCorrection,
  228. createdAt: event.timestamp,
  229. enteredBy: NigtscoutTreatment.local,
  230. bolus: nil,
  231. insulin: nil,
  232. notes: nil,
  233. carbs: Decimal(event.carbInput ?? 0),
  234. targetTop: nil,
  235. targetBottom: nil
  236. )
  237. default: return nil
  238. }
  239. }
  240. let uploaded = storage.retrieve(OpenAPS.Nightscout.uploadedPumphistory, as: [NigtscoutTreatment].self) ?? []
  241. let treatments = Array(Set([bolusesAndCarbs, temps].flatMap { $0 }).subtracting(Set(uploaded)))
  242. return treatments.sorted { $0.createdAt! > $1.createdAt! }
  243. }
  244. }