For reference, this is SR-7788 [and was brought up way back when in JSON Encoding / Decoding weird encoding of dictionary with enum values]
It's possible, though I'm not sure off the top of my head with which ABI effects, to extend Encodable
/Decodable
with marker requirements fullfilled by RawRepresentable
types:
extension Encodable {
func __stringRawValue() -> String? { nil }
}
extension Encodable where Self: RawRepresentable, Self.RawValue == String {
func __stringRawValue() -> String? { return self.rawValue }
}
enum Foo: Int, Encodable {
case a
}
enum Bar: String, Encodable {
case b
}
print(Foo.a.__stringRawValue() ?? "nil") // nil
print(Bar.b.__stringRawValue() ?? "nil") // b
It might be possible to take advantage of protocol extensions in this way (or other runtime trickery to make it happen), but...
... the big problem with making this change right now is that it's not backwards-compatible. Even if we enhance Dictionary.init(from:)
to be able to decode both forms (legacy unkeyed format, newer preferred keyed format), serialized payloads from newer versions of Swift wouldn't be decodable on older versions. For archived data, this is a serious concern.
There's unfortunately no good "switch" or option to enable somewhere to make it work. Specific encoders and decoders could override behavior for dictionaries to support this (so you could tell, for instance, JSONEncoder
to prefer converting RawRepresentable
keys to CodingKeys
), but this would be on an encoder-by-encoder basis; there's no hook we could add to Dictionary
specifically to easily turn this on or off.
There's also the practical matter to consider that could affect clients in a surprising way: if you take a type and make it RawRepresentable
by an Int
or String
in a later version of an app/framework, you can suddenly break your data format for yourself or others, with little recourse. Without having a good way to control the behavior, I think it would be too risky and breaking to enable this now.