PumpHistoryStorage.swift 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  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. try? storage.transaction { storage in
  155. try storage.append(events, to: file, uniqBy: \.id)
  156. uniqEvents = try storage.retrieve(file, as: [PumpHistoryEvent].self)
  157. .filter { $0.timestamp.addingTimeInterval(1.days.timeInterval) > Date() }
  158. .sorted { $0.timestamp > $1.timestamp }
  159. try 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. (try? 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. entededBy: NigtscoutTreatment.local,
  184. bolus: nil,
  185. insulin: nil,
  186. notes: nil,
  187. carbs: nil
  188. ))
  189. case .tempBasalDuration:
  190. if var last = result.popLast(), last.eventType == .nsTempBasal, last.createdAt == event.timestamp {
  191. last.duration = event.durationMin
  192. last.rawDuration = event
  193. result.append(last)
  194. }
  195. default: break
  196. }
  197. return result
  198. }
  199. let boluses = events.compactMap { event -> NigtscoutTreatment? in
  200. switch event.type {
  201. case .bolus:
  202. return NigtscoutTreatment(
  203. duration: event.duration,
  204. rawDuration: nil,
  205. rawRate: nil,
  206. absolute: nil,
  207. rate: nil,
  208. eventType: .bolus,
  209. createdAt: event.timestamp,
  210. entededBy: NigtscoutTreatment.local,
  211. bolus: event,
  212. insulin: event.amount,
  213. notes: nil,
  214. carbs: nil
  215. )
  216. default: return nil
  217. }
  218. }
  219. let uploaded = (try? storage.retrieve(OpenAPS.Nightscout.uploadedPumphistory, as: [NigtscoutTreatment].self)) ?? []
  220. let treatments = Array(Set([boluses, temps].flatMap { $0 }).subtracting(Set(uploaded)))
  221. return treatments.sorted { $0.createdAt! > $1.createdAt! }
  222. // .filter { $0.createdAt!.addingTimeInterval(3.hours.timeInterval) > Date() }
  223. }
  224. }