Skip to content

Commit 5e8ccad

Browse files
authored
Add Okio integration (Kotlin#1901)
* Add okio integration as separate module json-okio * Extract separate module json-tests that tests both Java streams and okio * Rewrite Java stream writer so that it uses much faster in-place UTF8 encoder instead of java.io.Writer * Add benchmarks for streams and okio * Disable targets for tests that are not supported by okio (they can't be run on LinuxX64-86 hosts anyway)
1 parent 605a35f commit 5e8ccad

File tree

436 files changed

+1243
-543
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

436 files changed

+1243
-543
lines changed

benchmark/build.gradle

+14-7
Original file line numberDiff line numberDiff line change
@@ -13,19 +13,26 @@ sourceCompatibility = 1.8
1313
targetCompatibility = 1.8
1414
jmh.jmhVersion = "1.22"
1515

16+
processJmhResources {
17+
doFirst {
18+
duplicatesStrategy(DuplicatesStrategy.EXCLUDE)
19+
}
20+
}
21+
1622
jmhJar {
17-
baseName 'benchmarks'
18-
classifier = null
19-
version = null
20-
destinationDir = file("$rootDir")
23+
archiveBaseName.set('benchmarks')
24+
archiveVersion.set('')
25+
destinationDirectory = file("$rootDir")
2126
}
2227

2328
dependencies {
2429
implementation 'org.openjdk.jmh:jmh-core:1.22'
25-
implementation 'com.google.guava:guava:24.1.1-jre'
26-
implementation 'com.fasterxml.jackson.core:jackson-databind:2.12.1'
27-
implementation 'com.fasterxml.jackson.module:jackson-module-kotlin:2.12.1'
30+
implementation 'com.google.guava:guava:31.1-jre'
31+
implementation 'com.fasterxml.jackson.core:jackson-databind:2.13.3'
32+
implementation 'com.fasterxml.jackson.module:jackson-module-kotlin:2.13.3'
33+
implementation "com.squareup.okio:okio:$okio_version"
2834
implementation project(':kotlinx-serialization-core')
2935
implementation project(':kotlinx-serialization-json')
36+
implementation project(':kotlinx-serialization-json-okio')
3037
implementation project(':kotlinx-serialization-protobuf')
3138
}

benchmark/src/jmh/kotlin/kotlinx/benchmarks/json/JacksonComparisonBenchmark.kt

+23
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,11 @@ import com.fasterxml.jackson.databind.*
44
import com.fasterxml.jackson.module.kotlin.*
55
import kotlinx.serialization.*
66
import kotlinx.serialization.json.*
7+
import kotlinx.serialization.json.okio.encodeToSink
8+
import okio.blackholeSink
9+
import okio.buffer
710
import org.openjdk.jmh.annotations.*
11+
import java.io.OutputStream
812
import java.util.concurrent.*
913

1014
@Warmup(iterations = 7, time = 1)
@@ -63,6 +67,13 @@ open class JacksonComparisonBenchmark {
6367
cookies = "_ga=GA1.2.971852807.1546968515"
6468
)
6569

70+
private val devNullSink = blackholeSink().buffer()
71+
private val devNullStream = object : OutputStream() {
72+
override fun write(b: Int) {}
73+
override fun write(b: ByteArray) {}
74+
override fun write(b: ByteArray, off: Int, len: Int) {}
75+
}
76+
6677
private val stringData = Json.encodeToString(DefaultPixelEvent.serializer(), data)
6778

6879
@Serializable
@@ -82,12 +93,24 @@ open class JacksonComparisonBenchmark {
8293
@Benchmark
8394
fun kotlinToString(): String = Json.encodeToString(DefaultPixelEvent.serializer(), data)
8495

96+
@Benchmark
97+
fun kotlinToStream() = Json.encodeToStream(DefaultPixelEvent.serializer(), data, devNullStream)
98+
99+
@Benchmark
100+
fun kotlinToOkio() = Json.encodeToSink(DefaultPixelEvent.serializer(), data, devNullSink)
101+
85102
@Benchmark
86103
fun kotlinToStringWithEscapes(): String = Json.encodeToString(DefaultPixelEvent.serializer(), dataWithEscapes)
87104

88105
@Benchmark
89106
fun kotlinSmallToString(): String = Json.encodeToString(SmallDataClass.serializer(), smallData)
90107

108+
@Benchmark
109+
fun kotlinSmallToStream() = Json.encodeToStream(SmallDataClass.serializer(), smallData, devNullStream)
110+
111+
@Benchmark
112+
fun kotlinSmallToOkio() = Json.encodeToSink(SmallDataClass.serializer(), smallData, devNullSink)
113+
91114
@Benchmark
92115
fun jacksonFromString(): DefaultPixelEvent = objectMapper.readValue(stringData, DefaultPixelEvent::class.java)
93116

benchmark/src/jmh/kotlin/kotlinx/benchmarks/json/TwitterBenchmark.kt

+10
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ package kotlinx.benchmarks.json
33
import kotlinx.benchmarks.model.*
44
import kotlinx.serialization.json.*
55
import org.openjdk.jmh.annotations.*
6+
import java.io.OutputStream
67
import java.util.concurrent.*
78

89
@Warmup(iterations = 7, time = 1)
@@ -24,6 +25,12 @@ open class TwitterBenchmark {
2425

2526
private val jsonImplicitNulls = Json { explicitNulls = false }
2627

28+
private val devNullStream = object : OutputStream() {
29+
override fun write(b: Int) {}
30+
override fun write(b: ByteArray) {}
31+
override fun write(b: ByteArray, off: Int, len: Int) {}
32+
}
33+
2734
@Setup
2835
fun init() {
2936
require(twitter == Json.decodeFromString(Twitter.serializer(), Json.encodeToString(Twitter.serializer(), twitter)))
@@ -38,4 +45,7 @@ open class TwitterBenchmark {
3845

3946
@Benchmark
4047
fun encodeTwitter() = Json.encodeToString(Twitter.serializer(), twitter)
48+
49+
@Benchmark
50+
fun encodeTwitterStream() = Json.encodeToStream(Twitter.serializer(), twitter, devNullStream)
4151
}

build.gradle

+3-1
Original file line numberDiff line numberDiff line change
@@ -144,6 +144,8 @@ allprojects {
144144
}
145145
}
146146

147+
def unpublishedProjects = ["benchmark", "guide", "kotlinx-serialization-json-tests"] as Set
148+
147149
subprojects {
148150
tasks.withType(org.jetbrains.kotlin.gradle.tasks.AbstractKotlinCompile).all { task ->
149151
if (task.name.contains("Test") || task.name.contains("Jmh")) {
@@ -155,7 +157,7 @@ subprojects {
155157

156158
apply from: rootProject.file('gradle/teamcity.gradle')
157159
// Configure publishing for some artifacts
158-
if (project.name != "benchmark" && project.name != "guide") {
160+
if (!unpublishedProjects.contains(project.name)) {
159161
apply from: rootProject.file('gradle/publishing.gradle')
160162
}
161163

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
public final class kotlinx/serialization/json/okio/OkioStreamsKt {
2+
public static final fun decodeFromSource (Lkotlinx/serialization/json/Json;Lkotlinx/serialization/DeserializationStrategy;Lokio/Source;)Ljava/lang/Object;
3+
public static final fun decodeSourceToSequence (Lkotlinx/serialization/json/Json;Lokio/Source;Lkotlinx/serialization/DeserializationStrategy;Lkotlinx/serialization/json/DecodeSequenceMode;)Lkotlin/sequences/Sequence;
4+
public static synthetic fun decodeSourceToSequence$default (Lkotlinx/serialization/json/Json;Lokio/Source;Lkotlinx/serialization/DeserializationStrategy;Lkotlinx/serialization/json/DecodeSequenceMode;ILjava/lang/Object;)Lkotlin/sequences/Sequence;
5+
public static final fun encodeToSink (Lkotlinx/serialization/json/Json;Lkotlinx/serialization/SerializationStrategy;Ljava/lang/Object;Lokio/Sink;)V
6+
}
7+

formats/json-okio/build.gradle.kts

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
/*
2+
* Copyright 2017-2022 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
3+
*/
4+
import Java9Modularity.configureJava9ModuleInfo
5+
6+
plugins {
7+
kotlin("multiplatform")
8+
kotlin("plugin.serialization")
9+
}
10+
11+
apply(from = rootProject.file("gradle/native-targets.gradle"))
12+
apply(from = rootProject.file("gradle/configure-source-sets.gradle"))
13+
14+
kotlin {
15+
sourceSets {
16+
val commonMain by getting {
17+
dependencies {
18+
api(project(":kotlinx-serialization-core"))
19+
api(project(":kotlinx-serialization-json"))
20+
implementation("com.squareup.okio:okio:${property("okio_version")}")
21+
}
22+
}
23+
val commonTest by getting {
24+
dependencies {
25+
implementation("com.squareup.okio:okio:${property("okio_version")}")
26+
}
27+
}
28+
}
29+
}
30+
31+
project.configureJava9ModuleInfo()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,139 @@
1+
/*
2+
* Copyright 2017-2022 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
3+
*/
4+
5+
package kotlinx.serialization.json.okio
6+
7+
import kotlinx.serialization.*
8+
import kotlinx.serialization.json.DecodeSequenceMode
9+
import kotlinx.serialization.json.Json
10+
import kotlinx.serialization.json.internal.*
11+
import kotlinx.serialization.json.okio.internal.JsonToOkioStreamWriter
12+
import kotlinx.serialization.json.internal.decodeToSequenceByReader
13+
import kotlinx.serialization.json.okio.internal.OkioSerialReader
14+
import okio.*
15+
16+
/**
17+
* Serializes the [value] with [serializer] into a [target] using JSON format and UTF-8 encoding.
18+
*
19+
* If [target] is not a [BufferedSink] then called [Sink.buffer] for it to create buffered wrapper.
20+
*
21+
* @throws [SerializationException] if the given value cannot be serialized to JSON.
22+
* @throws [okio.IOException] If an I/O error occurs and sink can't be written to.
23+
*/
24+
@ExperimentalSerializationApi
25+
public fun <T> Json.encodeToSink(
26+
serializer: SerializationStrategy<T>,
27+
value: T,
28+
target: Sink
29+
) {
30+
val buffered = if (target is BufferedSink) target else target.buffer()
31+
val writer = JsonToOkioStreamWriter(buffered)
32+
try {
33+
encodeByWriter(writer, serializer, value)
34+
} finally {
35+
writer.release()
36+
}
37+
}
38+
39+
/**
40+
* Serializes given [value] to a [target] using UTF-8 encoding and serializer retrieved from the reified type parameter.
41+
*
42+
* If [target] is not a [BufferedSink] then called [Sink.buffer] for it to create buffered wrapper.
43+
*
44+
* @throws [SerializationException] if the given value cannot be serialized to JSON.
45+
* @throws [okio.IOException] If an I/O error occurs and sink can't be written to.
46+
*/
47+
@ExperimentalSerializationApi
48+
public inline fun <reified T> Json.encodeToSink(
49+
value: T,
50+
target: Sink
51+
): Unit = encodeToSink(serializersModule.serializer(), value, target)
52+
53+
54+
/**
55+
* Deserializes JSON from [source] using UTF-8 encoding to a value of type [T] using [deserializer].
56+
*
57+
* If [source] is not a [BufferedSource] then called [Source.buffer] for it to create buffered wrapper.
58+
*
59+
* Note that this functions expects that exactly one object would be present in the source
60+
* and throws an exception if there are any dangling bytes after an object.
61+
*
62+
* @throws [SerializationException] if the given JSON input cannot be deserialized to the value of type [T].
63+
* @throws [okio.IOException] If an I/O error occurs and source can't be read from.
64+
*/
65+
@ExperimentalSerializationApi
66+
public fun <T> Json.decodeFromSource(
67+
deserializer: DeserializationStrategy<T>,
68+
source: Source
69+
): T {
70+
val buffered = if (source is BufferedSource) source else source.buffer()
71+
return decodeByReader(deserializer, OkioSerialReader(buffered))
72+
}
73+
74+
/**
75+
* Deserializes the contents of given [source] to the value of type [T] using UTF-8 encoding and
76+
* deserializer retrieved from the reified type parameter.
77+
*
78+
* If [source] is not a [BufferedSource] then called [Source.buffer] for it to create buffered wrapper.
79+
*
80+
* Note that this functions expects that exactly one object would be present in the stream
81+
* and throws an exception if there are any dangling bytes after an object.
82+
*
83+
* @throws [SerializationException] if the given JSON input cannot be deserialized to the value of type [T].
84+
* @throws [okio.IOException] If an I/O error occurs and source can't be read from.
85+
*/
86+
@ExperimentalSerializationApi
87+
public inline fun <reified T> Json.decodeFromSource(source: Source): T =
88+
decodeFromSource(serializersModule.serializer(), source)
89+
90+
91+
/**
92+
* Transforms the given [source] into lazily deserialized sequence of elements of type [T] using UTF-8 encoding and [deserializer].
93+
* Unlike [decodeFromSource], [source] is allowed to have more than one element, separated as [format] declares.
94+
*
95+
* If [source] is not a [BufferedSource] then called [Source.buffer] for it to create buffered wrapper.
96+
*
97+
* Elements must all be of type [T].
98+
* Elements are parsed lazily when resulting [Sequence] is evaluated.
99+
* Resulting sequence is tied to the stream and can be evaluated only once.
100+
*
101+
* **Resource caution:** this method neither closes the [source] when the parsing is finished nor provides a method to close it manually.
102+
* It is a caller responsibility to hold a reference to a stream and close it. Moreover, because stream is parsed lazily,
103+
* closing it before returned sequence is evaluated completely will result in [Exception] from decoder.
104+
*
105+
* @throws [SerializationException] if the given JSON input cannot be deserialized to the value of type [T].
106+
* @throws [okio.IOException] If an I/O error occurs and source can't be read from.
107+
*/
108+
@ExperimentalSerializationApi
109+
public fun <T> Json.decodeSourceToSequence(
110+
source: Source,
111+
deserializer: DeserializationStrategy<T>,
112+
format: DecodeSequenceMode = DecodeSequenceMode.AUTO_DETECT
113+
): Sequence<T> {
114+
val buffered = if (source is BufferedSource) source else source.buffer()
115+
return decodeToSequenceByReader(OkioSerialReader(buffered), deserializer, format)
116+
}
117+
118+
/**
119+
* Transforms the given [source] into lazily deserialized sequence of elements of type [T] using UTF-8 encoding and deserializer retrieved from the reified type parameter.
120+
* Unlike [decodeFromSource], [source] is allowed to have more than one element, separated as [format] declares.
121+
*
122+
* If [source] is not a [BufferedSource] then called [Source.buffer] for it to create buffered wrapper.
123+
*
124+
* Elements must all be of type [T].
125+
* Elements are parsed lazily when resulting [Sequence] is evaluated.
126+
* Resulting sequence is tied to the stream and constrained to be evaluated only once.
127+
*
128+
* **Resource caution:** this method does not close [source] when the parsing is finished neither provides method to close it manually.
129+
* It is a caller responsibility to hold a reference to a stream and close it. Moreover, because stream is parsed lazily,
130+
* closing it before returned sequence is evaluated fully would result in [Exception] from decoder.
131+
*
132+
* @throws [SerializationException] if the given JSON input cannot be deserialized to the value of type [T].
133+
* @throws [okio.IOException] If an I/O error occurs and source can't be read from.
134+
*/
135+
@ExperimentalSerializationApi
136+
public inline fun <reified T> Json.decodeSourceToSequence(
137+
source: Source,
138+
format: DecodeSequenceMode = DecodeSequenceMode.AUTO_DETECT
139+
): Sequence<T> = decodeSourceToSequence(source, serializersModule.serializer(), format)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
/*
2+
* Copyright 2017-2022 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
3+
*/
4+
5+
@file:Suppress("INVISIBLE_REFERENCE", "INVISIBLE_MEMBER")
6+
7+
package kotlinx.serialization.json.okio.internal
8+
9+
import kotlinx.serialization.json.internal.ESCAPE_STRINGS
10+
import kotlinx.serialization.json.internal.JsonWriter
11+
import kotlinx.serialization.json.internal.SerialReader
12+
import okio.*
13+
14+
internal class JsonToOkioStreamWriter(private val target: BufferedSink) : JsonWriter {
15+
override fun writeLong(value: Long) {
16+
write(value.toString())
17+
}
18+
19+
override fun writeChar(char: Char) {
20+
target.writeUtf8CodePoint(char.code)
21+
}
22+
23+
override fun write(text: String) {
24+
target.writeUtf8(text)
25+
}
26+
27+
override fun writeQuoted(text: String) {
28+
target.writeUtf8CodePoint('"'.code)
29+
var lastPos = 0
30+
for (i in text.indices) {
31+
val c = text[i].code
32+
if (c < ESCAPE_STRINGS.size && ESCAPE_STRINGS[c] != null) {
33+
target.writeUtf8(text, lastPos, i) // flush prev
34+
target.writeUtf8(ESCAPE_STRINGS[c]!!)
35+
lastPos = i + 1
36+
}
37+
}
38+
39+
if (lastPos != 0) target.writeUtf8(text, lastPos, text.length)
40+
else target.writeUtf8(text)
41+
target.writeUtf8CodePoint('"'.code)
42+
}
43+
44+
override fun release() {
45+
target.flush()
46+
}
47+
}
48+
49+
internal class OkioSerialReader(private val source: BufferedSource): SerialReader {
50+
override fun read(buffer: CharArray, bufferOffset: Int, count: Int): Int {
51+
var i = 0
52+
while (i < count && !source.exhausted()) {
53+
buffer[i] = source.readUtf8CodePoint().toChar()
54+
i++
55+
}
56+
return if (i > 0) i else -1
57+
}
58+
}
59+

formats/json-okio/gradle.properties

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
kotlin.mpp.enableCompatibilityMetadataVariant=false

0 commit comments

Comments
 (0)