commit | a368181feec40aa7d520a9669cd5d51d0c69f4f1 | [log] [tgz] |
---|---|---|
author | Leonid Startsev <sandwwraith@gmail.com> | Wed Aug 09 16:07:06 2017 +0300 |
committer | Leonid Startsev <sandwwraith@gmail.com> | Wed Aug 09 16:07:06 2017 +0300 |
tree | c26ef0d7635f540debc36dbc07eda1c1fb57e497 | |
parent | a587393822477bfa5a76f1b605dcf651932c5e55 [diff] |
Moved serializers resolving via reflection to separate file
Kotlin serialization plugin consists of three parts: a gradle compiler plugin, an IntelliJ plugin and a runtime library. This is the runtime library. To build any project with serialization (including this one), you'll need a serialization gradle plugin. To have a proper syntax highlighting in the Intellij IDEA, you'll need an IDEA plugin.
Runtime library provides basic classes:
Interfaces which are called by compiler-generated code (KInput
, KOutput
)
Basic skeleton implementations of these interfaces in which you should override some methods if you want to implement custom data format (ElementValueInput/Output
, NamedValueInput/Output
, ElementValueTransformer
)
Some internal classes like built-ins and collections serializers
Also, runtime library provides some ready-to-use serialization formats, see below.
Make sure you have serialization gradle plugin installed to your local maven repository. Then, run ./gradlew publishToMavenLocal
. After that, you can include this library in arbitrary projects like usual gradle dependency:
repositories { jcenter() mavenLocal() } dependencies { compile "org.jetbrains.kotlinx:serialization-runtime:1.1-SNAPSHOT" }
import kotlinx.serialization.* import kotlinx.serialization.json.JSON @Serializable data class Data(val a: Int, @Optional val b: String = "42") fun main(args: Array<String>) { println(JSON.stringify(Data(42))) // {"a": 42, "b": "42"} val obj = JSON.parse<Data>("""{"a":42}""") // Data(a=42, b="42") }
More complicated examples and examples of implementing custom formats can be found in examples
folder (work in progress). Detailed documentation located in DOC.md.
Runtime library provides three ready-to use formats: JSON, CBOR and ProtoBuf. Usage of the first two formats is pretty straightforward and obvious from the example above. Notes on them: because JSON doesn't support maps with keys other than strings (and primitives), Kotlin maps with non-trivial key types are serialized as JSON lists. CBOR doesn't have this limitation, and Kotlin maps are serialized as CBOR maps, but some parsers (like jackson-dataformat-cbor
) don't support this.
Because protobuf relies on serial ids of fields, called 'tags', you have to provide this information, using serial annotation @SerialId
:
@Serializable data class KTestInt32(@SerialId(1) val a: Int)
This class is equivalent to the following proto definition:
message Int32 { required int32 a = 1; }
Note that we are using proto2 semantics, where all fields are explicitly required or optional.
Number format is set via @ProtoType
annotation. ProtoNumberType.DEFAULT
is default varint encoding (intXX
), SIGNED
is signed ZigZag representation (sintXX
), and FIXED
is fixedXX
type. uintXX
and sfixedXX
are not supported yet.
Repeated fields represented as lists. Because format spec says that if the list is empty, there will be no elements in the stream with such tag, you must explicitly mark any filed of list type with @Optional
annotation with default = emptyList()
. Same for maps.
Other known issues and limitations:
Packed repeated fields are not supported
Unknows fields aren't skipped, exception is thrown instead
If fields with list tag are going in the arbitrary order, they are not merged into one list, they got overwritten instead.
More examples of mappings from proto definitions to Koltin classes can be found in test data: here and here