In my custom initializer I'd like to decode a dictionary from JSON and then assign its values to properties in the class. To my surprise, compiler does not allow me to decode the dictionary, I get the error:
Value of protocol type 'Any' cannot conform to 'Decodable'; only struct/enum/class types can conform to protocols
If I try to decode dictionary of type [String: Decodable] the error message says:
Value of protocol type 'Decodable' cannot conform to 'Decodable'; only struct/enum/class types can conform to protocols
My initializer looks like this:
public let total: Int
required init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
...
if let dict = try container.decodeIfPresent([String: Any].self, forKey: .tracks),
let value = dict["total"] as? Int { // Error is displayed at this line
total = value
} else {
total = 0
}
...
}
When I looked for the answer I found this answer and according to it the code above should not cause any problems.
What you are looking for is
nestedContainer
. It helps you "skip" a hierarchy level in your JSON. Ie: let's imagine that in your code, it's all in the same level (one struct), but in the JSON, it's not, there is a sub dictionary.For test purpose, your JSON might look like:
If we want in our model:
We need to use
nestedContainer(keyedBy: forKey:)
:In your sample, you used
decodeIfPresent()
for the subdictionary. It's unclear if it was for test purpose, or if the sub dictionary was sometimes not present. If that's the case and you can have a JSON like this:Then, before calling
nestedContainer(keyedBy: forKey:)
, useif container.contains(TopLevelKeys.tracks) {}
and set default values if needed in theelse
case.