Как получить доступ к ответу json в SwiftUI? - PullRequest
2 голосов
/ 25 марта 2020

Итак, я делаю HTTP-запрос и получаю ответ, который выглядит следующим образом:

enter image description here

Как получить «bogdanp» в виде строки и как получить доступ к свойствам объекта? Например, как получить доступ к «имени» или «телефону» и т. Д. c.?

Вот мой HTTP-код:

func getPatients(link: String, username: String)
{
    var code = 0 // this is the variable that stores the error code from the response, it is initially set to 0
    let parameters: [String : Any] = ["username": username]
    let url = URL(string: link)!
    let session = URLSession.shared
    var request = URLRequest(url: url)

    request.httpMethod = "POST"
    do {
        request.httpBody = try JSONSerialization.data(withJSONObject: parameters, options: .prettyPrinted)
    } catch _ {
        }

        request.addValue("application/json", forHTTPHeaderField: "Content-Type")
        request.addValue("application/json", forHTTPHeaderField: "Accept")

        let task = session.dataTask(with: request as URLRequest, completionHandler: { data, response, error in
            guard error == nil else {
                return
            }
            guard let data = data else {
                return
            }
            do {
                if let jsonResponse = try JSONSerialization.jsonObject(with: data, options: .mutableContainers) as? [String: Any] {
                        // Here I want to access the json response
                    }
            } catch _ {
            }
        })
    task.resume()
}

1 Ответ

2 голосов
/ 25 марта 2020

Лучший способ сделать запрос вызова с SwiftUI - это использовать Combine.

Сначала вам нужно будет создать модель вашего JSON объекта, который вы хотите получить обратно:

import Foundation

//MARK: - Your object to retrieve from JSON
struct Doctor: Codable, Identifiable {
  let id = UUID()
  let patients: [Patients]
}

struct Patients: Codable {
  let id: String
  let name: String
  let phone: String
}

Затем вы создадите класс, который будет обрабатывать ваши JSON запросы, используя Объедините (я добавил плюс для вас, чтобы обработать любую ошибку ответа):

import Foundation
import Combine

class Network {

  // Handle your request errors
  enum Error: LocalizedError {
    case invalidResponse
    case addressUnreachable(URL)

    var errorDescription: String? {
      switch self {
      case .invalidResponse:
        return "The server responded with garbage."
      case .addressUnreachable(let url):
        return "\(url.absoluteString) is unreachable."
      }
    }
  }

  // Add your url
  let urlRequest = URL(string: "your url")!

  // Networking on concurrent queue
  let networkQueue = DispatchQueue(label: "Networking",
                                   qos: .default,
                                   attributes: .concurrent)

  // Combine network call (This replace your previous code)
  func downloadPatients() -> AnyPublisher<Doctor, Error> {
    URLSession.shared
      .dataTaskPublisher(for: urlRequest)
      .receive(on: networkQueue)
      .map(\.data)
      .decode(type: Doctor.self, decoder: JSONDecoder())
      .mapError { (error) -> Network.Error in
        switch error {
        case is URLError:
          return Error.addressUnreachable(self.urlRequest)
        default:
          return Error.invalidResponse
        }
    }
    .eraseToAnyPublisher()
  }
}

И теперь, в вашем файле SwiftUI, где вам нужны эти значения, вам просто нужно вызвать функцию downloadPatients () и использовать возвращаемые данные по мере того как вы sh:

import SwiftUI

let networkRequest = Network()

//MARK: - Call this function where you want to make your call
func loadPatients() {
  _ = networkRequest.downloadPatients()
    .sink(
      receiveCompletion: {
        print("Received Completion: \($0)") },
      receiveValue: { doctor in
        // doctor is your response and [0].name is your first patient name
        print(doctor.patients[0].name) }
  )
}
...