NightscoutAPI.swift 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  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 = 2
  13. static let timeout: TimeInterval = 2
  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! JSONEncoder().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. ]
  85. if let date = sinceDate {
  86. let dateItem = URLQueryItem(
  87. name: "find[created_at][$gte]",
  88. value: Formatter.iso8601withFractionalSeconds.string(from: date)
  89. )
  90. components.queryItems?.append(dateItem)
  91. }
  92. var request = URLRequest(url: components.url!)
  93. request.allowsConstrainedNetworkAccess = false
  94. request.timeoutInterval = Config.timeout
  95. if let secret = secret {
  96. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  97. }
  98. return service.run(request)
  99. .retry(Config.retryCount)
  100. .decode(type: [CarbsEntry].self, decoder: JSONCoding.decoder)
  101. .eraseToAnyPublisher()
  102. }
  103. func fetchTempTargets(sinceDate: Date? = nil) -> AnyPublisher<[TempTarget], Swift.Error> {
  104. var components = URLComponents()
  105. components.scheme = url.scheme
  106. components.host = url.host
  107. components.port = url.port
  108. components.path = Config.treatmentsPath
  109. components.queryItems = [URLQueryItem(name: "find[eventType]", value: "Temporary+Target")]
  110. if let date = sinceDate {
  111. let dateItem = URLQueryItem(
  112. name: "find[created_at][$gte]",
  113. value: Formatter.iso8601withFractionalSeconds.string(from: date)
  114. )
  115. components.queryItems?.append(dateItem)
  116. }
  117. var request = URLRequest(url: components.url!)
  118. request.allowsConstrainedNetworkAccess = false
  119. request.timeoutInterval = Config.timeout
  120. if let secret = secret {
  121. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  122. }
  123. return service.run(request)
  124. .retry(Config.retryCount)
  125. .decode(type: [TempTarget].self, decoder: JSONCoding.decoder)
  126. .eraseToAnyPublisher()
  127. }
  128. }
  129. private extension String {
  130. func sha1() -> String {
  131. let data = Data(utf8)
  132. var digest = [UInt8](repeating: 0, count: Int(CC_SHA1_DIGEST_LENGTH))
  133. data.withUnsafeBytes {
  134. _ = CC_SHA1($0.baseAddress, CC_LONG(data.count), &digest)
  135. }
  136. let hexBytes = digest.map { String(format: "%02hhx", $0) }
  137. return hexBytes.joined()
  138. }
  139. }