Разбор Stock API, который содержит даты в качестве ключей, используя протокол Codable - PullRequest
2 голосов
/ 07 июня 2019

Пытаюсь разобрать стоковые API от Alpha Vantage.Вот как выглядит ответ: Демонстрация ответа API

Я настроил четыре класса для использования при декодировании и кодировании:

  1. Акции
  2. Метаданные
  3. Временной ряд
  4. Открыть Высокий Низкий Закрыть Закрыть

Я думаю, что проблема заключается в классах временных рядов, так как я предполагаю, что получаю даты массивав качестве ключа, каждый из которых содержит значения Open, Close, High, Low.Все три класса соответствуют протоколу Codable. Я изменил значение ключей в перечислении, чтобы оно могло соответствовать правильному JSON Response.

    var stocks = [Stocks]()
override func viewDidLoad() {
    super.viewDidLoad()
    view.backgroundColor = .red
    loadURL()
}


func loadURL(){
    let stocksURL = URL(string: "https://www.alphavantage.co/query?function=TIME_SERIES_INTRADAY&symbol=MSFT&interval=5min&apikey=demo")
    print("Inside loadURL")
    guard let url = stocksURL else { return }
    if let data = try? Data(contentsOf: url){
        print("Inside data")
        parse(json: data)
    }
}

func parse(json: Data){
    let jsonDecoder = JSONDecoder()
    print("before Decoding")
    if let json = try? jsonDecoder.decode([Stocks].self, from: json){
        print("Inside parsing")
        stocks = json
        print(stocks[0].meta_data.symbol)
        print(stocks[0].time_series.OHLC[0].open)
    }
}
}

class Stocks: Codable {
var meta_data: MetaData
var time_series: TimeSeries

enum CodingKeys: String, CodingKey {
    case meta_data   = "Meta Data"
    case time_series = "Time Series (5min)"
}
}

class MetaData: Codable {
var information: String
var symbol: String
var lastRefreshed: String
var outputSize: String
var timeZone: String

enum CodingKeys: String, CodingKey {
    case information   = "1. Information"
    case symbol        = "2. Symbol"
    case lastRefreshed = "3. Last Refreshed"
    case outputSize    = "4. Output Size"
    case timeZone      = "5. Time Zone"
}
}

class TimeSeries: Codable {
var OHLC: [OpenHighLowClose]
}

class OpenHighLowClose: Codable {
var open: Double
var high: Double
var low: Double
var close: Double
var volume: Double

enum CodingKeys: String, CodingKey {
    case open   = "1. open"
    case high   = "2. high"
    case low    = "3. low"
    case close  = "4. close"
    case volume = "5. volume"
}
}

1 Ответ

1 голос
/ 07 июня 2019

Вот правильные типы данных с вашим ответом.

// MARK: - Response
struct Response: Codable {
    let metaData: MetaData
    let timeSeries5Min: [String: TimeSeries5Min]

    enum CodingKeys: String, CodingKey {
        case metaData = "Meta Data"
        case timeSeries5Min = "Time Series (5min)"
    }
}

// MARK: - MetaData
struct MetaData: Codable {
    let the1Information, the2Symbol, the3LastRefreshed, the4Interval: String
    let the5OutputSize, the6TimeZone: String

    enum CodingKeys: String, CodingKey {
        case the1Information = "1. Information"
        case the2Symbol = "2. Symbol"
        case the3LastRefreshed = "3. Last Refreshed"
        case the4Interval = "4. Interval"
        case the5OutputSize = "5. Output Size"
        case the6TimeZone = "6. Time Zone"
    }
}

// MARK: - TimeSeries5Min
struct TimeSeries5Min: Codable {
    let the1Open, the2High, the3Low, the4Close: String
    let the5Volume: String

    enum CodingKeys: String, CodingKey {
        case the1Open = "1. open"
        case the2High = "2. high"
        case the3Low = "3. low"
        case the4Close = "4. close"
        case the5Volume = "5. volume"
    }
}

Использование:

  func loadURL() {
    let stocksURL = URL(string: "https://www.alphavantage.co/query?function=TIME_SERIES_INTRADAY&symbol=MSFT&interval=5min&apikey=demo")
    URLSession.shared.dataTask(with: stocksURL!) { (data, response, error) in
      if let error = error {
        print(error)
        return
      }
      do {
        let response = try JSONDecoder().decode(Response.self, from: data!)
        response.timeSeries5Min.forEach({ (keyValue) in
          print(keyValue)
        })
      } catch {
        print(error)
      }
    }.resume()
  }
...