NightscoutAPI.swift 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  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 retryCount = 5
  12. }
  13. enum Error: LocalizedError {
  14. case badStatusCode
  15. case missingURL
  16. }
  17. let url: URL
  18. let secret: String?
  19. private let service = NetworkService()
  20. }
  21. extension NightscoutAPI {
  22. func checkConnection() -> AnyPublisher<Void, Swift.Error> {
  23. struct Check: Codable, Equatable {
  24. var eventType = "Note"
  25. var enteredBy = "feeaps-x://"
  26. var notes = "FreeAPS X connected"
  27. }
  28. let check = Check()
  29. var request = URLRequest(url: url.appendingPathComponent("api/v1/treatments.json"))
  30. request.httpMethod = "POST"
  31. request.addValue("application/json", forHTTPHeaderField: "Content-Type")
  32. if let secret = secret {
  33. request.addValue(secret.sha1(), forHTTPHeaderField: "api-secret")
  34. }
  35. request.httpBody = try! JSONEncoder().encode(check)
  36. return service.run(request)
  37. .map { _ in () }
  38. .eraseToAnyPublisher()
  39. }
  40. func fetchLast(_ count: Int) -> AnyPublisher<[BloodGlucose], Swift.Error> {
  41. var components = URLComponents()
  42. components.scheme = url.scheme
  43. components.host = url.host
  44. components.port = url.port
  45. components.path = Config.entriesPath
  46. components.queryItems = [URLQueryItem(name: "count", value: "\(count)")]
  47. var request = URLRequest(url: components.url!)
  48. request.allowsConstrainedNetworkAccess = false
  49. return URLSession.shared.dataTaskPublisher(for: request)
  50. .retry(Config.retryCount)
  51. .tryMap { output in
  52. guard let response = output.response as? HTTPURLResponse, response.statusCode == 200 else {
  53. throw Error.badStatusCode
  54. }
  55. return output.data
  56. }
  57. .decode(type: [BloodGlucose].self, decoder: JSONCoding.decoder)
  58. .map {
  59. $0.filter { $0.isStateValid }
  60. .map {
  61. var reading = $0
  62. reading.glucose = $0.sgv
  63. return reading
  64. }
  65. }
  66. .eraseToAnyPublisher()
  67. }
  68. }
  69. private extension String {
  70. func sha1() -> String {
  71. let data = Data(utf8)
  72. var digest = [UInt8](repeating: 0, count: Int(CC_SHA1_DIGEST_LENGTH))
  73. data.withUnsafeBytes {
  74. _ = CC_SHA1($0.baseAddress, CC_LONG(data.count), &digest)
  75. }
  76. let hexBytes = digest.map { String(format: "%02hhx", $0) }
  77. return hexBytes.joined()
  78. }
  79. }