интернированые адаптеры
This commit is contained in:
parent
a5668b57e6
commit
e01afbefe7
@ -42,6 +42,8 @@ import ru.dbotthepony.kstarbound.io.*
|
||||
import ru.dbotthepony.kstarbound.io.json.AABBTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.AABBiTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.EitherTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.InternedJsonElementAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.InternedStringAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.Vector2dTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.Vector2fTypeAdapter
|
||||
import ru.dbotthepony.kstarbound.io.json.Vector2iTypeAdapter
|
||||
@ -105,25 +107,14 @@ class Starbound : ISBFileLocator {
|
||||
setFieldNamingPolicy(FieldNamingPolicy.IDENTITY)
|
||||
setPrettyPrinting()
|
||||
|
||||
// чтоб строки всегда intern'ились
|
||||
registerTypeAdapter(object : TypeAdapter<String>() {
|
||||
override fun write(out: JsonWriter, value: String?) {
|
||||
if (value == null)
|
||||
out.nullValue()
|
||||
else
|
||||
out.value(value)
|
||||
}
|
||||
|
||||
override fun read(`in`: JsonReader): String? {
|
||||
return stringInterner.intern(TypeAdapters.STRING.read(`in`) ?: return null)
|
||||
}
|
||||
})
|
||||
registerTypeAdapter(InternedStringAdapter(stringInterner))
|
||||
registerTypeAdapter(InternedJsonElementAdapter(stringInterner))
|
||||
|
||||
// Обработчик @JsonImplementation
|
||||
registerTypeAdapterFactory(JsonImplementationTypeFactory)
|
||||
|
||||
// ImmutableList, ImmutableSet, ImmutableMap
|
||||
registerTypeAdapterFactory(ImmutableCollectionAdapterFactory)
|
||||
registerTypeAdapterFactory(ImmutableCollectionAdapterFactory(stringInterner))
|
||||
|
||||
// ArrayList
|
||||
registerTypeAdapterFactory(ArrayListAdapterFactory)
|
||||
|
@ -0,0 +1,63 @@
|
||||
package ru.dbotthepony.kstarbound.io.json
|
||||
|
||||
import com.google.common.collect.Interner
|
||||
import com.google.gson.JsonArray
|
||||
import com.google.gson.JsonElement
|
||||
import com.google.gson.JsonNull
|
||||
import com.google.gson.JsonObject
|
||||
import com.google.gson.JsonPrimitive
|
||||
import com.google.gson.TypeAdapter
|
||||
import com.google.gson.internal.LazilyParsedNumber
|
||||
import com.google.gson.internal.bind.TypeAdapters
|
||||
import com.google.gson.stream.JsonReader
|
||||
import com.google.gson.stream.JsonToken
|
||||
import com.google.gson.stream.JsonWriter
|
||||
|
||||
class InternedJsonElementAdapter(val stringInterner: Interner<String>) : TypeAdapter<JsonElement>() {
|
||||
private val _true = JsonPrimitive(true)
|
||||
private val _false = JsonPrimitive(false)
|
||||
|
||||
override fun write(out: JsonWriter, value: JsonElement?) {
|
||||
return TypeAdapters.JSON_ELEMENT.write(out, value)
|
||||
}
|
||||
|
||||
override fun read(`in`: JsonReader): JsonElement? {
|
||||
return when (val p = `in`.peek()) {
|
||||
JsonToken.STRING -> JsonPrimitive(stringInterner.intern(`in`.nextString()))
|
||||
JsonToken.NUMBER -> JsonPrimitive(LazilyParsedNumber(`in`.nextString()))
|
||||
JsonToken.BOOLEAN -> if (`in`.nextBoolean()) _true else _false
|
||||
JsonToken.NULL -> JsonNull.INSTANCE
|
||||
JsonToken.BEGIN_ARRAY -> {
|
||||
val output = JsonArray()
|
||||
`in`.beginArray()
|
||||
while (`in`.hasNext()) { output.add(read(`in`)) }
|
||||
`in`.endArray()
|
||||
output
|
||||
}
|
||||
JsonToken.BEGIN_OBJECT -> {
|
||||
val output = JsonObject()
|
||||
`in`.beginObject()
|
||||
while (`in`.hasNext()) { output.add(stringInterner.intern(`in`.nextName()), read(`in`)) }
|
||||
`in`.endObject()
|
||||
output
|
||||
}
|
||||
else -> throw IllegalArgumentException(p.toString())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class InternedStringAdapter(val stringInterner: Interner<String>) : TypeAdapter<String>() {
|
||||
override fun write(out: JsonWriter, value: String?) {
|
||||
out.value(value)
|
||||
}
|
||||
|
||||
override fun read(`in`: JsonReader): String? {
|
||||
if (`in`.peek() == JsonToken.NULL)
|
||||
return null
|
||||
|
||||
if (`in`.peek() == JsonToken.BOOLEAN)
|
||||
return `in`.nextBoolean().toString()
|
||||
|
||||
return stringInterner.intern(`in`.nextString())
|
||||
}
|
||||
}
|
@ -3,6 +3,7 @@ package ru.dbotthepony.kstarbound.io.json.factory
|
||||
import com.google.common.collect.ImmutableList
|
||||
import com.google.common.collect.ImmutableMap
|
||||
import com.google.common.collect.ImmutableSet
|
||||
import com.google.common.collect.Interner
|
||||
import com.google.gson.Gson
|
||||
import com.google.gson.TypeAdapter
|
||||
import com.google.gson.TypeAdapterFactory
|
||||
@ -10,7 +11,7 @@ import com.google.gson.internal.`$Gson$Types`
|
||||
import com.google.gson.reflect.TypeToken
|
||||
|
||||
@Suppress("unchecked_cast")
|
||||
object ImmutableCollectionAdapterFactory : TypeAdapterFactory {
|
||||
class ImmutableCollectionAdapterFactory(val stringInterner: Interner<String> = Interner { it }) : TypeAdapterFactory {
|
||||
override fun <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
|
||||
when (type.rawType) {
|
||||
ImmutableList::class.java -> {
|
||||
@ -27,7 +28,7 @@ object ImmutableCollectionAdapterFactory : TypeAdapterFactory {
|
||||
val (elementType0, elementType1) = `$Gson$Types`.getMapKeyAndValueTypes(type.type, type.rawType)
|
||||
|
||||
if (`$Gson$Types`.getRawType(elementType0) == String::class.java) {
|
||||
return ImmutableMapTypeAdapter(gson.getAdapter(TypeToken.get(elementType1)))as TypeAdapter<T>
|
||||
return ImmutableMapTypeAdapter(stringInterner, gson.getAdapter(TypeToken.get(elementType1))) as TypeAdapter<T>
|
||||
}
|
||||
|
||||
return ImmutableArrayMapTypeAdapter(
|
||||
|
@ -2,13 +2,14 @@ package ru.dbotthepony.kstarbound.io.json.factory
|
||||
|
||||
import com.google.common.collect.ImmutableList
|
||||
import com.google.common.collect.ImmutableMap
|
||||
import com.google.common.collect.Interner
|
||||
import com.google.gson.JsonSyntaxException
|
||||
import com.google.gson.TypeAdapter
|
||||
import com.google.gson.stream.JsonReader
|
||||
import com.google.gson.stream.JsonToken
|
||||
import com.google.gson.stream.JsonWriter
|
||||
|
||||
class ImmutableMapTypeAdapter<V>(val elementAdapter: TypeAdapter<V>) : TypeAdapter<ImmutableMap<String, V>>() {
|
||||
class ImmutableMapTypeAdapter<V>(val stringInterner: Interner<String>, val elementAdapter: TypeAdapter<V>) : TypeAdapter<ImmutableMap<String, V>>() {
|
||||
override fun write(out: JsonWriter, value: ImmutableMap<String, V>?) {
|
||||
if (value == null) {
|
||||
out.nullValue()
|
||||
@ -29,12 +30,25 @@ class ImmutableMapTypeAdapter<V>(val elementAdapter: TypeAdapter<V>) : TypeAdapt
|
||||
if (reader.peek() == JsonToken.NULL)
|
||||
return null
|
||||
|
||||
if (reader.peek() == JsonToken.BEGIN_ARRAY) {
|
||||
val builder = ImmutableMap.Builder<String, V>()
|
||||
|
||||
while (reader.peek() !== JsonToken.END_ARRAY) {
|
||||
builder.put(
|
||||
stringInterner.intern(reader.nextString()),
|
||||
elementAdapter.read(reader) ?: throw JsonSyntaxException("Nulls are not allowed, near ${reader.path}")
|
||||
)
|
||||
}
|
||||
|
||||
return builder.build()
|
||||
}
|
||||
|
||||
reader.beginObject()
|
||||
|
||||
val builder = ImmutableMap.Builder<String, V>()
|
||||
|
||||
while (reader.peek() != JsonToken.END_OBJECT) {
|
||||
builder.put(reader.nextName(), elementAdapter.read(reader) ?: throw JsonSyntaxException("Nulls are not allowed, near ${reader.path}"))
|
||||
while (reader.peek() !== JsonToken.END_OBJECT) {
|
||||
builder.put(stringInterner.intern(reader.nextName()), elementAdapter.read(reader) ?: throw JsonSyntaxException("Nulls are not allowed, near ${reader.path}"))
|
||||
}
|
||||
|
||||
reader.endObject()
|
||||
|
Loading…
Reference in New Issue
Block a user