kotlinx.serialization.cbor.CborDecodingException: ожидалось начало карты, но найдено 78

Я пытался использовать один из пакета сериализации Kotlin со следующим настраиваемым примером класса. Я расширю этот класс в соответствии с реальным вариантом использования, добавив List реального класса (который также необходимо сериализовать), но сейчас проблема просто заключается в десериализации. Мне удалось заставить сериализацию работать (во втором фрагменте кода есть как объект, который нужно сериализовать, так и результирующий String), и кажется, что процесс десериализации требует того же способом (т. е. с использованием экземпляра NullableSerializer) (на самом деле, я не нашел прямо сейчас простого и правильного способа десериализации закодированного объекта. Вот класс, простой, но сложный для подражания.

@Serializable(with = EhEH.Companion::class)
data class EhEH(
    val i: Int
    , val txt: String
    //, val list: Array<String> // next feature, also with complex data, not only "simple" Strings
) {

    @Serializer(forClass = EhEH::class)
    companion object : KSerializer<EhEH> {
        public var deserialEhEH: DeserializationStrategy<EhEH> = EhEH


        override val descriptor: SerialDescriptor =
            SerialDescriptor("EhEHSerializer", PrimitiveKind.STRING)

        override fun serialize(encoder: Encoder, obj: EhEH) =
            encoder.encodeString(obj::class.qualifiedName!!)

        override fun deserialize(decoder: Decoder): EhEH {
            val dec = decoder.beginStructure(descriptor)
            var txt: String? = null
            var x: Int? = null
            var i = -1
            var notFinished = true
            do {
                i = dec.decodeElementIndex(descriptor)
                when (i) {
                    CompositeDecoder.READ_DONE -> notFinished = false
                    0 -> x = dec.decodeStringElement(descriptor, i).toIntOrNull()
                    1 -> txt = dec.decodeStringElement(descriptor, i)
                    else -> throw SerializationException("Unknown index $i")
                }
            } while (notFinished)
            dec.endStructure(descriptor)
            return EhEH(
                x ?: throw MissingFieldException("x"),
                txt ?: throw MissingFieldException("txt")
            )
        }

    }

    override fun toString(): String {
        return "EhEH(i=$i, s='$txt')"
    }
}

Функция ручного тестирования (значение которой просто печатается):

@InternalSerializationApi
    fun testCborString(): String {
        var e: EhEH = EhEH(
            7,
            "poreccio"
            //, listOf("just", "another", "vacuum", "test")
        )
        return Cbor.dumps(
            NullableSerializer(
                EhEH.Companion
            ), e
        ) + " <-> " + (
                {
                    (Cbor.load(
                        NullableSerializer(
                            EhEH.Companion
                        ),
                        Cbor.dump(
                            NullableSerializer(EhEH.Companion), e
                        )
                        //the dumped value should be "781d62632e7472797669756d2e6170692e726573706f6e7365732e45684548"
                    )
                            as EhEH).toString()
                } as () -> String)()
    }

Когда я пытаюсь запустить его, исключение составляет вызов Cbor.load:

kotlinx.serialization.cbor.CborDecodingException: Expected start of map, but found 78

Я единственный во ВСЕМ Интернете, у кого есть эта проблема?


person Marco Ottina    schedule 03.06.2020    source источник


Ответы (1)


Я нашел способ самостоятельно, ошибок было в основном две:

  1. переменная override val descriptor: SerialDescriptor была настроена для распознавания String, но это явно не тот параметр, который нужен, потому что это класс:

    // val stringArraySerializer= ArraySerializer(String.serializer())
    SerialDescriptor("EhEHSerializer", StructureKind.CLASS) {
            element<Int>("i")
            element<String>("txt")
            // element<Array<String>>("list") // kept here for future references for Array of primitive stuffs
            // element("arrComplexClass", AComplexClassTaggedSerializable.descriptor) // in case of Array of complex objects, a tag @Serializable and a companion object implementing Kserializable<AComplexClassTaggedSerializable> is needed
        }
    
  2. аналогично был реализован метод override fun serialize(encoder: Encoder, obj: EhEH) для создания String (в частности, строки, содержащей имя класса). Требуемая реализация:

    override fun serialize(encoder: Encoder, obj: EhEH){
        var ce = encoder.beginStructure(
            descriptor,
            Int.serializer(),
            String.serializer(),
            //stringArraySerializer, // kept here for future references for
            // serializerArray_AComplexClassTaggedSerializable
        )
        ce.encodeIntElement(descriptor, 0, obj.i)
        ce.encodeStringElement(descriptor, 1, obj.txt)
        /*ce.encodeSerializableElement(
            descriptor, 2,
            stringArraySerializer, obj.arrOfStrings
        )*/
        /*ce.encodeSerializableElement(
            descriptor, 3,
            serializerArray_AComplexClassTaggedSerializable,
            obj.arrComplexClass
        )*/
        ce.endStructure(descriptor)
    
    }
    

    Также другие фундаментальные примечания:

  3. требуется для переноса сложных объектов на NullableSerializer, также в случае Array из них:

    val serializerArray_AComplexClassTaggedSerializable =
        ArraySerializer<AComplexClassTaggedSerializable, AComplexClassTaggedSerializable?>(
            NullableSerializer(
                AComplexClassTaggedSerializable.Companion
            )
        ) // it's a variable used on the code above
    
  4. после сериализации и десериализации (т.е. Cbor.dump и Cbor.load соответственно) в качестве первого параметра требуется NullableSerializer:

    var nse = NullableSerializer(AComplexClassTaggedSerializable)
    println( Cbor.dump(nse, someInstanceOfThatComplexClass ).toString())
    //and....
    var o = (Cbor.load(nse, "...here go something..") as EhEH
    // use "o"
    
person Marco Ottina    schedule 05.06.2020