NightscoutAPI.swift 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. import Combine
  2. import CommonCrypto
  3. import Foundation
  4. class NightscoutAPI {
  5. init(url: URL, secret: String? = nil) {
  6. self.url = url
  7. self.secret = secret
  8. }
  9. private enum Config {
  10. static let entriesPath = "/api/v1/entries/sgv.json"
  11. static let treatmentsPath = "/api/v1/treatments.json"
  12. static let retryCount = 1
  13. static let timeout: TimeInterval = 30
  14. }
  15. enum Error: LocalizedError {
  16. case badStatusCode
  17. case missingURL
  18. }
  19. let url: URL
  20. let secret: String?
  21. private let service = NetworkService()
  22. }
  23. extension NightscoutAPI {
  24. func checkConnection() -> AnyPublisher<Void, Swift.Error> {
  25. struct Check: Codable, Equatable {
  26. var eventType = "Note"
  27. var enteredBy = "feeaps-x://"
  28. var notes = "FreeAPS X connected"
  29. }
  30. let check = Check()
  31. var request = URLRequest(url: url.appendingPathComponent(Config.treatmentsPath))
  32. request.httpMethod = "POST"
  33. request.addValue("application/json", forHTTPHeaderField: "Content-Type")
  34. if let secret = secret {
  35. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  36. }
  37. request.httpBody = try! JSONCoding.encoder.encode(check)
  38. return service.run(request)
  39. .map { _ in () }
  40. .eraseToAnyPublisher()
  41. }
  42. func fetchLastGlucose(_ count: Int, sinceDate: Date? = nil) -> AnyPublisher<[BloodGlucose], Swift.Error> {
  43. var components = URLComponents()
  44. components.scheme = url.scheme
  45. components.host = url.host
  46. components.port = url.port
  47. components.path = Config.entriesPath
  48. components.queryItems = [URLQueryItem(name: "count", value: "\(count)")]
  49. if let date = sinceDate {
  50. let dateItem = URLQueryItem(
  51. name: "find[dateString][$gte]",
  52. value: Formatter.iso8601withFractionalSeconds.string(from: date)
  53. )
  54. components.queryItems?.append(dateItem)
  55. }
  56. var request = URLRequest(url: components.url!)
  57. request.allowsConstrainedNetworkAccess = false
  58. request.timeoutInterval = Config.timeout
  59. if let secret = secret {
  60. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  61. }
  62. return service.run(request)
  63. .retry(Config.retryCount)
  64. .decode(type: [BloodGlucose].self, decoder: JSONCoding.decoder)
  65. .map {
  66. $0.filter { $0.isStateValid }
  67. .map {
  68. var reading = $0
  69. reading.glucose = $0.sgv
  70. return reading
  71. }
  72. }
  73. .eraseToAnyPublisher()
  74. }
  75. func fetchCarbs(sinceDate: Date? = nil) -> AnyPublisher<[CarbsEntry], Swift.Error> {
  76. var components = URLComponents()
  77. components.scheme = url.scheme
  78. components.host = url.host
  79. components.port = url.port
  80. components.path = Config.treatmentsPath
  81. components.queryItems = [
  82. URLQueryItem(name: "find[carbs][$exists]", value: "true"),
  83. URLQueryItem(name: "find[enteredBy][$ne]", value: CarbsEntry.manual),
  84. URLQueryItem(name: "find[enteredBy][$ne]", value: NigtscoutTreatment.local)
  85. ]
  86. if let date = sinceDate {
  87. let dateItem = URLQueryItem(
  88. name: "find[created_at][$gte]",
  89. value: Formatter.iso8601withFractionalSeconds.string(from: date)
  90. )
  91. components.queryItems?.append(dateItem)
  92. }
  93. var request = URLRequest(url: components.url!)
  94. request.allowsConstrainedNetworkAccess = false
  95. request.timeoutInterval = Config.timeout
  96. if let secret = secret {
  97. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  98. }
  99. return service.run(request)
  100. .retry(Config.retryCount)
  101. .decode(type: [CarbsEntry].self, decoder: JSONCoding.decoder)
  102. .eraseToAnyPublisher()
  103. }
  104. func fetchTempTargets(sinceDate: Date? = nil) -> AnyPublisher<[TempTarget], Swift.Error> {
  105. var components = URLComponents()
  106. components.scheme = url.scheme
  107. components.host = url.host
  108. components.port = url.port
  109. components.path = Config.treatmentsPath
  110. components.queryItems = [
  111. URLQueryItem(name: "find[eventType]", value: "Temporary+Target"),
  112. URLQueryItem(name: "find[enteredBy][$ne]", value: TempTarget.manual)
  113. ]
  114. if let date = sinceDate {
  115. let dateItem = URLQueryItem(
  116. name: "find[created_at][$gte]",
  117. value: Formatter.iso8601withFractionalSeconds.string(from: date)
  118. )
  119. components.queryItems?.append(dateItem)
  120. }
  121. var request = URLRequest(url: components.url!)
  122. request.allowsConstrainedNetworkAccess = false
  123. request.timeoutInterval = Config.timeout
  124. if let secret = secret {
  125. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  126. }
  127. return service.run(request)
  128. .retry(Config.retryCount)
  129. .decode(type: [TempTarget].self, decoder: JSONCoding.decoder)
  130. .eraseToAnyPublisher()
  131. }
  132. func fetchAnnouncement(sinceDate: Date? = nil) -> AnyPublisher<[Announcement], Swift.Error> {
  133. var components = URLComponents()
  134. components.scheme = url.scheme
  135. components.host = url.host
  136. components.port = url.port
  137. components.path = Config.treatmentsPath
  138. components.queryItems = [
  139. URLQueryItem(name: "find[eventType]", value: "Announcement"),
  140. URLQueryItem(name: "find[enteredBy]", value: Announcement.remote)
  141. ]
  142. if let date = sinceDate {
  143. let dateItem = URLQueryItem(
  144. name: "find[created_at][$gte]",
  145. value: Formatter.iso8601withFractionalSeconds.string(from: date)
  146. )
  147. components.queryItems?.append(dateItem)
  148. }
  149. var request = URLRequest(url: components.url!)
  150. request.allowsConstrainedNetworkAccess = false
  151. request.timeoutInterval = Config.timeout
  152. if let secret = secret {
  153. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  154. }
  155. return service.run(request)
  156. .retry(Config.retryCount)
  157. .decode(type: [Announcement].self, decoder: JSONCoding.decoder)
  158. .eraseToAnyPublisher()
  159. }
  160. func uploadTreatments(_ treatments: [NigtscoutTreatment]) -> AnyPublisher<Void, Swift.Error> {
  161. var components = URLComponents()
  162. components.scheme = url.scheme
  163. components.host = url.host
  164. components.port = url.port
  165. components.path = Config.treatmentsPath
  166. var request = URLRequest(url: components.url!)
  167. request.allowsConstrainedNetworkAccess = false
  168. request.timeoutInterval = Config.timeout
  169. request.addValue("application/json", forHTTPHeaderField: "Content-Type")
  170. if let secret = secret {
  171. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  172. }
  173. request.httpBody = try! JSONCoding.encoder.encode(treatments)
  174. request.httpMethod = "POST"
  175. return service.run(request)
  176. .retry(Config.retryCount)
  177. .map { _ in () }
  178. .eraseToAnyPublisher()
  179. }
  180. }
  181. private extension String {
  182. func sha1() -> String {
  183. let data = Data(utf8)
  184. var digest = [UInt8](repeating: 0, count: Int(CC_SHA1_DIGEST_LENGTH))
  185. data.withUnsafeBytes {
  186. _ = CC_SHA1($0.baseAddress, CC_LONG(data.count), &digest)
  187. }
  188. let hexBytes = digest.map { String(format: "%02hhx", $0) }
  189. return hexBytes.joined()
  190. }
  191. }