I fully disagree with @BenjaminBriggs and @feldur. FlatBuffer, Protobuf, Thrift, Cap'n proto, etc. can't really be addressed by this library feature . This wouldn't work on many levels.
Schema DSLs
Schemas aren't defined in Swift. They aren't defined in C. They aren't defined in Java, JavaScript, Haskell, Python, etc.
All aforementioned formats have designed their own schema language as a means of sharing that information between different ecosystems. Disparate ecosystems don't usually have a way of sharing a single library that defines serialization formats, so instead they share a .proto file which will generate said library for every and any language you would want to use.
This is an incredible thing to have. A service written in Go can use the same wire data model as the one written in JavaScript.
We aren't getting rid of external schema files anytime soon.
I mean, you don't necessarily have to use a schema file
You of course can "port" schema file to swift.
import Thrift
@ThriftCodable
struct Vec3 {
@ThriftTag(1) var x: Int32
@ThriftTag(2) var y: Int32
@ThriftTag(3) var z: Int32
}
@ThriftCodable
enum Color: Int32 {
case Red = 1
case Blue
case Green
}
@ThriftCodable
struct Monster {
@ThriftTag(1) var pos: Vec3?
@ThriftTag(2) var mana: Int16 = 150
@ThriftTag(3) var hp: Int16 = 100
@ThriftTag(4) var name: String?
@available(*, deprecated)
@ThriftTag(5) var friendly = false
@ThriftTag(6) var inventory: [UInt8]?
@ThriftTag(7) var color: Color = .Blue
}
It is useful for cases where only swift systems would need to use this serialization format. Or where you are fine with hand-porting a small one-off definition, just to avoid adopting the whole code generation tooling.
As I would discuss later though, this is does not fit well with, nor does it require this proposal. One can implement such a package today, and consume it with SwiftPM with no trouble.
There's also JSON/XML schemas
Same argument can be applied to standardized JSON schemas. You can use them in the same way to share definitions between languages. You'd also would then inherit all the complexity of code generation :(
It's usually less prevalent, since you can use JSON without a schema, but you cannot(-ish) do the same with say Cap'n proto. Also JSON is human readable and schemas are often inferred by looking at the output, which is a useful property.
This is what Swift OpenAPI generator basically does.
So do we want to have an ability to
import "./definitions.proto"
?
Reliance on third-party tooling
But like I said, .proto
isn't .swift
. Do we now want to teach the swift compiler how to parse them? Does the Foundation need to provide a generic ProtobufDecoder
? What about Thrift? FlatBuffers?
These are a really big projects. It is an enormous burden to integrate or port them into the compiler infrastructure. It's a non-starter.
What we can do is integrate existing third-party code-generators into the macro system.. maybe..
Pipe dreams... I mean, macros. Yes, macros!
One could imagine installing a swift-flatbuffers
package, add something like this:
import FlatBuffers
#Definition("""
namespace MyGame;
attribute "priority";
enum Color : byte { Red = 1, Green, Blue }
struct Vec3 {
x:float;
y:float;
z:float;
}
table Monster {
pos:Vec3;
mana:short = 150;
hp:short = 100;
name:string;
friendly:bool = false (deprecated, priority: 1);
inventory:[ubyte];
color:Color = Blue;
}
root_type Monster;
""")
let monster = Monster(
pos: Vec3(x: 0, y: 420, z: 69),
name: "Cthulhu",
inventory: [],
color: .Red
)
serialize(monster)
And that is possible. It could simplify things. There's no need to deal with flatc
, no autogenerated // PLESE DON'T EDIT
files to manage. There are downsides of course.
Ideally we would've standardized something like #embed
a long time ago to allow for a much better:
import FlatBuffers
#Definition(#embed("./monster.fbs"))
(God I wish #embed
/#fileLiteral
/#whatever
was a thing [1][2])
Interoperability with format-agnostic types
Thrift and Protobuf require explicit tags. For a good reason.
How would that play out when serializing a format-agnostic type, like CGRect
for e.g. There would be no benefit for those formats to support the proposed (codenamed CommonCodable
) stdlib mechanism.
All in all they are a vastly different beasts . Even in-between themselves.
I don't think we can do anything to ease adoption of these serialization formats. As you may see, no other language out there attempts to do so either. It is a good thing that this was stated as an explicit non-goal, and it makes sense.
What we can do, is provide a common interchange data model for commonly used self-describing formats, like JSON, CBOR, plist, msgpack, YAML, TOML, java properties, ini, XML (not sure about that one tbh), SQL(ite) rows, etc.
We can provide common format decoders (JSON, plist) built-into swift's distribution (ie. Foundation).
We can provide a way to integrate commonly used serialization data-model for use with less prominent data formats, and let users use them via separate packages.
Codable
did that for the most part. We can do so better!
These are worthy goals to achieve, even if we can't fit Protobuf into it. I don't think it can be. I'd really like to be proven wrong though.