0

I am trying to decode this JSON from this url https://jsonodds.com/api/test/odds

Sometimes some of the values are nil and I wanted to avoid making all of my properties optional so I am trying to implement custom decoding.

[
    {
        "ID": "e83bf033-dbde-49ac-a073-d385ebdc66d1",
        "HomeTeam": "Xavier",
        "AwayTeam": "Marquette",
        "Sport": 2,
        "MatchTime": "2018-01-24T23:30:00",
        "Odds": [
            {
                "ID": "50742eed-a1c8-4958-8678-50627ffb665e",
                "EventID": "e83bf033-dbde-49ac-a073-d385ebdc66d1",
                "MoneyLineHome": "0",
                "MoneyLineAway": "0",
                "PointSpreadHome": "-4.0",
                "PointSpreadAway": "4.0",
                "PointSpreadHomeLine": "-110",
                "PointSpreadAwayLine": "-110",
                "TotalNumber": "78.0",
                "OverLine": "-115",
                "UnderLine": "-105",
                "DrawLine": "0",
                "LastUpdated": "2018-01-24T20:50:17",
                "SiteID": 3,
                "OddType": "First Half"
            },
            {
                "ID": "e83bf033-dbde-49ac-a073-d385ebdc66d1",
                "EventID": "e83bf033-dbde-49ac-a073-d385ebdc66d1",
                "MoneyLineHome": "-305",
                "MoneyLineAway": "255",
                "PointSpreadHome": "-7.0",
                "PointSpreadAway": "7.0",
                "PointSpreadHomeLine": "-112",
                "PointSpreadAwayLine": "-108",
                "TotalNumber": "162.5",
                "OverLine": "-104",
                "UnderLine": "-116",
                "DrawLine": "0",
                "LastUpdated": "2018-01-24T20:50:17",
                "SiteID": 3,
                "OddType": "Game"
            }
        ],
        "Details": "Marquette at Xavier",
        "HomeROT": "720",
        "AwayROT": "719"
    },

I currently have this but I don't know if I am accounting for when the values are nil, and also am struggling to make it work since the "Odds" key in the JSON is an array, so I was not sure how to go about doing that. I only need the "Odds" for when the "OddType" in the JSON is equal to "Game". I was just gonna filter that out after I got all of the data from the JSON, since I wasn't sure if there was a way to do this while parsing the JSON.

struct GameInformation: Codable {    
    let homeTeam: String
    let awayTeam: String
    let sport: Int
    let matchTime: String
    let odds: [Odds]
    let details: String
}

struct Odds: Codable {
    let moneyLineHome: String
    let moneyLineAway: String
    let pointSpreadHome: String
    let pointSpreadAway: String
    let pointSpreadHomeLine: String
    let pointSpreadAwayLine: String
    let totalNumber: String
    let overLine: String
    let underLine: String
    let drawLine: String
    let lastUpdated: String
    let oddType: String
}

extension Odds {
    enum CodingKeys: String, CodingKey {
        case moneyLineHome = "MoneyLineHome"
        case moneyLineAway = "MoneyLineAway"
        case pointSpreadHome = "PointSpreadHome"
        case pointSpreadAway = "PointSpreadAway"
        case pointSpreadHomeLine = "PointSpreadHomeLine"
        case pointSpreadAwayLine = "PointSpreadAwayLine"
        case totalNumber = "TotalNumber"
        case overLine = "OverLine"
        case underLine = "UnderLine"
        case drawLine = "DrawLine"
        case lastUpdated = "LastUpdated"
        case oddType = "OddType"
    }

    enum OddsKey: String, CodingKey {
        case odds = "Odds"
    }

    init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: OddsKey.self)
        let oddsContainer = try container.nestedContainer(keyedBy: CodingKeys.self, forKey: .odds)

        moneyLineHome = try oddsContainer.decodeIfPresent(String.self, forKey: .moneyLineHome) ?? "N/A"
        moneyLineAway = try oddsContainer.decodeIfPresent(String.self, forKey: .moneyLineAway) ?? "N/A"
        pointSpreadHome = try oddsContainer.decodeIfPresent(String.self, forKey: .pointSpreadHome) ?? "N/A"
        pointSpreadAway = try oddsContainer.decodeIfPresent(String.self, forKey: .pointSpreadAway) ?? "N/A"
        pointSpreadHomeLine = try oddsContainer.decodeIfPresent(String.self, forKey: .pointSpreadHomeLine) ?? "N/A"
        pointSpreadAwayLine = try oddsContainer.decodeIfPresent(String.self, forKey: .pointSpreadAwayLine) ?? "N/A"
        totalNumber = try oddsContainer.decodeIfPresent(String.self, forKey: .totalNumber) ?? "N/A"
        overLine = try oddsContainer.decodeIfPresent(String.self, forKey: .overLine) ?? "N/A"
        underLine = try oddsContainer.decodeIfPresent(String.self, forKey: .underLine) ?? "N/A"
        drawLine = try oddsContainer.decodeIfPresent(String.self, forKey: .drawLine) ?? "N/A"
        lastUpdated = try oddsContainer.decodeIfPresent(String.self, forKey: .lastUpdated) ?? "N/A"
        oddType = try oddsContainer.decodeIfPresent(String.self, forKey: .oddType) ?? "N/A"
    }
}

extension GameInformation {
    enum CodingKeys: String, CodingKey {
        case homeTeam = "HomeTeam"
        case awayTeam = "AwayTeam"
        case sport = "Sport"
        case matchTime = "MatchTime"
        case odds = "Odds"
        case details = "Details"
    }

    init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        homeTeam = try container.decodeIfPresent(String.self, forKey: .homeTeam) ?? "N/A"
        awayTeam = try container.decodeIfPresent(String.self, forKey: .awayTeam) ?? "N/A"
        sport = try container.decodeIfPresent(Int.self, forKey: .sport) ?? 20
        matchTime = try container.decodeIfPresent(String.self, forKey: .matchTime) ?? "N/A"
        details = try container.decodeIfPresent(String.self, forKey: .details) ?? "N/A"
        odds = [try Odds(from: decoder)]
    }
}

Any help is much appreciated!

1 Answer 1

0

Try setting odds like this in the GameInformation init:

odds = try container.decode(Array<Odds>.self, forKey: .odds)

Next you should remove OddsKey enum and update Odds init:

init(from decoder: Decoder) throws {
    let container = try decoder.container(keyedBy: CodingKeys.self)

    moneyLineHome = try container.decodeIfPresent(String.self, forKey: .moneyLineHome) ?? "N/A"
    moneyLineAway = try container.decodeIfPresent(String.self, forKey: .moneyLineAway) ?? "N/A"
    pointSpreadHome = try container.decodeIfPresent(String.self, forKey: .pointSpreadHome) ?? "N/A"
    pointSpreadAway = try container.decodeIfPresent(String.self, forKey: .pointSpreadAway) ?? "N/A"
    pointSpreadHomeLine = try container.decodeIfPresent(String.self, forKey: .pointSpreadHomeLine) ?? "N/A"
    pointSpreadAwayLine = try container.decodeIfPresent(String.self, forKey: .pointSpreadAwayLine) ?? "N/A"
    totalNumber = try container.decodeIfPresent(String.self, forKey: .totalNumber) ?? "N/A"
    overLine = try container.decodeIfPresent(String.self, forKey: .overLine) ?? "N/A"
    underLine = try container.decodeIfPresent(String.self, forKey: .underLine) ?? "N/A"
    drawLine = try container.decodeIfPresent(String.self, forKey: .drawLine) ?? "N/A"
    lastUpdated = try container.decodeIfPresent(String.self, forKey: .lastUpdated) ?? "N/A"
    oddType = try container.decodeIfPresent(String.self, forKey: .oddType) ?? "N/A"
}

I haven't tested it, but it should work. The thing is, we should parse Odds as a single object inside it's init, and as a collection inside of GameInformation init. I hope it makes sense.

I think it's better to filter after parsing, you can initialize odds like this in the GameInformation init (but it just doesn't seem right without utilizing optionals):

odds = try container.decode(Array<Odds>.self, forKey: .odds).filter {
        let mirror = Mirror(reflecting: $0)
        return !mirror.children.contains(where: { _, value in
            if let v = value as? String, v == "N/A" {
                return true
            } else {
                return false
            }
        })
    }
Sign up to request clarification or add additional context in comments.

2 Comments

That worked, thank you so much. Is there anyway to filter out objects that have "N/A" as one of their properties while parsing the JSON, or is the best way to go about that after all the JSON is parsed?
I added one way you could filter odds in my answer.

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.