Кодируемые типы HAL JSON

У кого-нибудь есть процесс для управления данными JSON типа HAL? Проблема, с которой я сталкиваюсь, заключается в том, что все запросы данных будут возвращать контейнер, который встраивает его фактический тип в ключ _embedded. Я изо всех сил пытаюсь понять, как декодировать этот тип, поскольку каждому встроенному ключу может быть назначено несколько [Any HalTypes]. Например, если я сделаю запрос на пункт меню или категорию меню, он вернет ту же всеобъемлющую структуру. Приведенный ниже JSON предназначен для категории меню.

E.g.,

Модель

// This file was generated from JSON Schema using quicktype, do not modify it directly.
// To parse the JSON, add this file to your project and do:
//
//   let category = try? newJSONDecoder().decode(Category.self, from: jsonData)

import Foundation

// MARK: - Category
struct Category: Codable {
    var embedded: Embedded?
    var links: CategoryLinksClass?
    var count, limit: Int?

    enum CodingKeys: String, CodingKey {
        case embedded = "_embedded"
        case links = "_links"
        case count, limit
    }
}

// MARK: - Embedded
struct Embedded: Codable {
    var categories: [CategoryElement]?
}

// MARK: - CategoryElement
struct CategoryElement: Codable {
    var links: CategoryLinks?
    var id: String?
    var level: Int?
    var name, posid: String?

    enum CodingKeys: String, CodingKey {
        case links = "_links"
        case id, level, name
        case posid = "pos_id"
    }
}

// MARK: - CategoryLinks
struct CategoryLinks: Codable {
    var linksSelf: Next?

    enum CodingKeys: String, CodingKey {
        case linksSelf = "self"
    }
}

// MARK: - Next
struct Next: Codable {
    var href: String?
    var type: String?
}

// MARK: - CategoryLinksClass
struct CategoryLinksClass: Codable {
    var next, linksSelf: Next?

    enum CodingKeys: String, CodingKey {
        case next
        case linksSelf = "self"
    }
}

JSON

{
  "_embedded": {
    "categories": [
      {
        "_links": {
          "self": {
            "href": "https://api.omnivore.io/1.0/locations/iE7e78GT/menu/categories/1001/",
            "type": "application/json; name=menu_category"
          }
        },
        "id": "1001",
        "level": 0,
        "name": "Entree",
        "pos_id": "1001"
      },
      {
        "_links": {
          "self": {
            "href": "https://api.omnivore.io/1.0/locations/iE7e78GT/menu/categories/1002/",
            "type": "application/json; name=menu_category"
          }
        },
        "id": "1002",
        "level": 0,
        "name": "Appetizer",
        "pos_id": "1002"
      }
    ]
  },
  "_links": {
    "next": {
      "href": "https://api.omnivore.io/1.0/locations/iE7e78GT/menu/categories/?limit=2&start=2",
      "type": "application/json; name=menu_category_list"
    },
    "self": {
      "href": "https://api.omnivore.io/1.0/locations/iE7e78GT/menu/categories/?limit=2",
      "type": "application/json; name=menu_category_list"
    }
  },
  "count": 2,
  "limit": 2
}

person dmwatson0101    schedule 28.07.2020    source источник


Ответы (1)


Вы можете сделать так, чтобы ключ _embedded каждый раз принимал общую структуру Codable вместо определенного типа.

struct Category<T: Codable>: Codable {
    var embedded: T?
    var links: CategoryLinksClass?
    var count, limit: Int?

    enum CodingKeys: String, CodingKey {
        case embedded = "_embedded"
        case links = "_links"
        case count, limit
    }
}

И создайте разные модели, которые возвращаются для ключа "_embedded".

struct Embedded: Codable { ... }

struct Menu: Codable { ... }

А затем укажите тип модели во время декодирования следующим образом:

do { let decoded = JSONDecoder().decode(Category<Embedded>.self, from: data) 
} catch { print(error) }
person Frankenstein    schedule 28.07.2020
comment
Спасибо попробуем вечером! - person dmwatson0101; 28.07.2020