У меня проблема с сериализацией JSON kotlinx и с аннотацией @Serializer на Android. Проблема возникает только на определенных уровнях API - кажется, работает выше уровня API 23 и выдает эту ошибку ниже:
java.lang.VerifyError: Rejecting class org.kethereum.wallet.model.KdfSerializer because it failed compile-time verification (declaration of 'org.kethereum.wallet.model.KdfSerializer' appears in /data/app/package.name-1/base.apk:classes3.dex)
at org.kethereum.wallet.model.WalletCrypto.write$Self(Wallet.kt)
at org.kethereum.wallet.model.WalletCrypto$$serializer.save(Wallet.kt)
at org.kethereum.wallet.model.WalletCrypto$$serializer.save(Wallet.kt:15)
at kotlinx.serialization.KOutput.writeSerializableValue(Serialization.kt:146)
at kotlinx.serialization.KOutput.writeSerializableElementValue(Serialization.kt:191)
at org.kethereum.wallet.model.Wallet.write$Self(Wallet.kt)
at org.kethereum.wallet.model.Wallet$$serializer.save(Wallet.kt)
at org.kethereum.wallet.model.Wallet$$serializer.save(Wallet.kt:53)
at kotlinx.serialization.KOutput.write(Serialization.kt:99)
at kotlinx.serialization.json.JSON.stringify(JSON.kt:40)
at kotlinx.serialization.json.JSON$Companion.stringify(JSON.kt:57)
at org.kethereum.wallet.WalletFileKt.generateWalletFile(WalletFile.kt:30)
Я нашел обходной путь для своего варианта использования - но хотел бы понять проблему.
При этом:
@Serializable
data class WalletCrypto(
var cipher: String,
var ciphertext: String,
var cipherparams: CipherParams,
var kdf: String,
@Serializable(with = KdfSerializer::class)
var kdfparams: KdfParams,
var mac: String) {
object KdfSerializer : KSerializer<KdfParams> {
override val serialClassDesc = SerialClassDescImpl("KDFSerializer")
override fun load(input: KInput) = throw NotImplementedError("loading WalletCrypto is not implemented - use WalletCryptoForImport instead")
override fun save(output: KOutput, obj: KdfParams) = when (obj) {
is ScryptKdfParams -> output.write(obj)
is Aes128CtrKdfParams -> output.write(obj)
}
}
}
вместо:
@Serializable
data class WalletCrypto(
var cipher: String,
var ciphertext: String,
var cipherparams: CipherParams,
var kdf: String,
@Serializable(with = KdfSerializer::class)
var kdfparams: KdfParams,
var mac: String) {
@Serializer(forClass = KdfParams::class)
object KdfSerializer : KSerializer<KdfParams> {
override fun save(output: KOutput, obj: KdfParams) = when (obj) {
is ScryptKdfParams -> output.write(obj)
is Aes128CtrKdfParams -> output.write(obj)
}
}
}
проблема исчезла. Таким образом, проблема, кажется, коренится в использовании аннотации @Serializer. KdfParams - это закрытый класс:
sealed class KdfParams {
abstract var dklen: Int
abstract var salt: String?
}