
commonMain.kotlinx.serialization.internal.Enums.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlinx-serialization-core-jvm Show documentation
Show all versions of kotlinx-serialization-core-jvm Show documentation
Kotlin multiplatform serialization runtime library
/*
* Copyright 2017-2020 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
*/
package kotlinx.serialization.internal
import kotlinx.serialization.*
import kotlinx.serialization.descriptors.*
import kotlinx.serialization.encoding.*
/*
* Descriptor used for explicitly serializable enums by the plugin.
* Designed to be consistent with `EnumSerializer.descriptor` and weird plugin usage.
*/
@Suppress("unused") // Used by the plugin
@PublishedApi
@OptIn(ExperimentalSerializationApi::class)
internal class EnumDescriptor(
name: String,
elementsCount: Int
) : PluginGeneratedSerialDescriptor(name, elementsCount = elementsCount) {
override val kind: SerialKind = SerialKind.ENUM
private val elementDescriptors by lazy {
Array(elementsCount) { buildSerialDescriptor(name + "." + getElementName(it), StructureKind.OBJECT) }
}
override fun getElementDescriptor(index: Int): SerialDescriptor = elementDescriptors.getChecked(index)
override fun equals(other: Any?): Boolean {
if (this === other) return true
if (other == null) return false
if (other !is SerialDescriptor) return false
if (other.kind !== SerialKind.ENUM) return false
if (serialName != other.serialName) return false
if (cachedSerialNames() != other.cachedSerialNames()) return false
return true
}
override fun toString(): String {
return elementNames.joinToString(", ", "$serialName(", ")")
}
override fun hashCode(): Int {
var result = serialName.hashCode()
val elementsHashCode = elementNames.elementsHashCodeBy { it }
result = 31 * result + elementsHashCode
return result
}
}
@PublishedApi
internal fun > createSimpleEnumSerializer(serialName: String, values: Array): KSerializer {
return EnumSerializer(serialName, values)
}
/**
* The function has a bug (#2121) and should not be used by new (1.8.20+) plugins. It is preserved for backward compatibility with previously compiled enum classes.
*/
@PublishedApi
internal fun > createMarkedEnumSerializer(
serialName: String,
values: Array,
names: Array,
annotations: Array?>
): KSerializer {
val descriptor = EnumDescriptor(serialName, values.size)
values.forEachIndexed { i, v ->
val elementName = names.getOrNull(i) ?: v.name
descriptor.addElement(elementName)
annotations.getOrNull(i)?.forEach {
descriptor.pushAnnotation(it)
}
}
return EnumSerializer(serialName, values, descriptor)
}
@PublishedApi
internal fun > createAnnotatedEnumSerializer(
serialName: String,
values: Array,
names: Array,
entryAnnotations: Array?>,
classAnnotations: Array?
): KSerializer {
val descriptor = EnumDescriptor(serialName, values.size)
classAnnotations?.forEach {
descriptor.pushClassAnnotation(it)
}
values.forEachIndexed { i, v ->
val elementName = names.getOrNull(i) ?: v.name
descriptor.addElement(elementName)
entryAnnotations.getOrNull(i)?.forEach {
descriptor.pushAnnotation(it)
}
}
return EnumSerializer(serialName, values, descriptor)
}
@PublishedApi
@OptIn(ExperimentalSerializationApi::class)
internal class EnumSerializer>(
serialName: String,
private val values: Array
) : KSerializer {
private var overriddenDescriptor: SerialDescriptor? = null
internal constructor(serialName: String, values: Array, descriptor: SerialDescriptor) : this(serialName, values) {
overriddenDescriptor = descriptor
}
override val descriptor: SerialDescriptor by lazy {
overriddenDescriptor ?: createUnmarkedDescriptor(serialName)
}
private fun createUnmarkedDescriptor(serialName: String): SerialDescriptor {
val d = EnumDescriptor(serialName, values.size)
values.forEach { d.addElement(it.name) }
return d
}
override fun serialize(encoder: Encoder, value: T) {
val index = values.indexOf(value)
if (index == -1) {
throw SerializationException(
"$value is not a valid enum ${descriptor.serialName}, " +
"must be one of ${values.contentToString()}"
)
}
encoder.encodeEnum(descriptor, index)
}
override fun deserialize(decoder: Decoder): T {
val index = decoder.decodeEnum(descriptor)
if (index !in values.indices) {
throw SerializationException(
"$index is not among valid ${descriptor.serialName} enum values, " +
"values size is ${values.size}"
)
}
return values[index]
}
override fun toString(): String = "kotlinx.serialization.internal.EnumSerializer<${descriptor.serialName}>"
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy