Skip to content

Latest commit

 

History

History
291 lines (215 loc) · 10.6 KB

README.md

File metadata and controls

291 lines (215 loc) · 10.6 KB

English | 简体中文

License: Apache License Version 2.0 Language Maven Central

Kudos

Kudos is short for Kotlin utilities for deserializing objects. It is designed to make it safer and easier to deserializing Kotlin classes with Gson and Jackson.

Background

When parsing JSON using common JSON serialization frameworks, Kotlin developers often face issues with no-arg constructors and property null safety. Let's illustrate these issues with some examples.

data class User(val name: String) {
    val firstName by lazy {
        name.split(" ").first()
    }
    val lastName = name.split(" ").last()
}

The data class User does not have a default no-arg constructor. When using a framework like Gson to parse the following JSON text:

{"name": "Benny Huo"}

Gson will create an instance of User using Unsafe because User lacks a no-arg constructor, which means that the constructor of User won't be invoked correctly. In other words, firstName and lastName will not be initialized correctly. This is actually very risky. Some frameworks, like Jackson, will throw an error when they find that User lacks a no-arg constructor, refusing to deserialize it.

To address this issue, Kotlin provides the NoArg plugin, which generates a default no-arg constructor for types annotated with a specific annotation.

// build.gradle.kts
plugins {
    kotlin("plugin.noarg") version "$kotlinVersion"
}

noArg {
    annotation("com.kanyun.annotations.PoKo")
}

// User.kt
@PoKo
data class User(val name: String) {
    val firstName by lazy {
        name.split(" ").first()
    }
    val lastName = name.split(" ").last()
}

We generate a no-arg constructor for classes annotated with @PoKo. In this case, User will have a constructor that simply calls the parent class's no-arg constructor. This constructor has an empty body, so firstName and lastName will still not be initialized correctly.

// build.gradle.kts
noArg {
    ...
    invokeInitializers = true
}

Fortunately, the NoArg plugin also provides a option called invokeInitializers. By default, this option is disabled. When you enable it, the generated no-arg constructor can correctly initialize the firstName property. However, the bad news is that when the constructor tries to initialize lastName, it will throw a NullPointerException because name has not been initialized at that point.

Another common issue is dealing with default values for parameters in the primary constructor. For example:

@PoKo
data class User(
    val id: Long, 
    val name: String,
    val age: Int = -1,
    val tel: String = ""
)

Not all User instances provide their own age and phone number, so we expect that when parsing, if the JSON does not contain the corresponding fields, the default values should be used.

JSON text:

{"id": 12, "name":  "Benny Huo"}

The result:

User(id=12, name=Benny Huo, age=0, tel=null)

This exposes two problems: one is that the default values for parameters in the primary constructor are completely ignored, and the other is that the non-nullable tel property is set to null, completely undermining type safety.

For these reasons, we typically recommend using Moshi or kotlinx.serialization.

However, migrating to these frameworks is not easy. kotlinx.serialization does not support Java. Moshi, while supporting Java, still has some differences in detail compared to Gson.

Furthermore, Moshi is not always faster than Gson as benchmarks said. Most benchmark tests we have seen totally ignored the initialization time of Moshi. When using Moshi-codegen based on KAPT/KSP, Moshi often takes more time to initialize because it creates a dedicated JsonAdapter for each class. We were surprised to find that using Moshi to parse JSON took 2-3 times longer than using Gson during our app's startup time optimization.

We kept thinking, is there a way to provide null safety and default values support of the primary constructor parameters for frameworks like Gson? The answer is Kudos.

Quick Start

1. Add the plugin to classpath

// Option 1
// The classic way, add the following code to build.gradle.kts in the root directory
buildscript {
    repositories {
        mavenCentral()
        // for snapshots
        maven("https://oss.sonatype.org/content/repositories/snapshots/")
    }
    dependencies {
        classpath("com.kanyun.kudos:kudos-gradle-plugin:$latest_version")
    }
}

subprojects {
    repositories {
        mavenCentral()
        // for snapshots
        maven("https://oss.sonatype.org/content/repositories/snapshots/")
    }
}

// Option 2
// The new way, add the following code to settings.gradle.kts
pluginManagement {
    repositories {
        mavenCentral()
        // for snapshots
        maven("https://oss.sonatype.org/content/repositories/snapshots/")
    }
    plugins {
        id("com.kanyun.kudos") version "$latest_version" apply false
    }
}

dependencyResolutionManagement {
    repositories {
        mavenCentral()
        // for snapshots
        maven("https://oss.sonatype.org/content/repositories/snapshots/")
    }
}

2. Apply the plugin

plugins {
    // Apply Kudos plugin. 
    // An enhanced no-arg constructor will be generated for classes annotated with @Kudos.
    id("com.kanyun.kudos")
}

kudos {
    // Enable Kudos.Gson. Generate @JsonAdapter for kudos classes and add kudos-gson to dependencies.
    gson = true
    // Enable Kudos.Jackson. Add kudos-jackson to dependencies.
    jackson = true
}

Dependencies below will be added when the plugins are applied.

com.kanyun.kudos:kudos-annotations
com.kanyun.kudos:kudos-runtime

// Only for Kudos.Gson
com.kanyun.kudos:kudos-gson

// Only for Kudos.Jackson
com.kanyun.kudos:kudos-jackson

You can add these libraries to your dependencies explicitly if needed.

3. Annotate classes with @Kudos

For types that need to add Kudos parsing support, simply add the @Kudos annotation, for example:

@Kudos
data class User(
    val id: Long, 
    val name: String,
    val age: Int = -1,
    val tel: String = ""
)

After compilation, it is roughly equivalent to:

@Kudos
// If the 'com.kanyun.kudos.gson' plugin is enabled, the @JsonAdapter annotation is generated
@JsonAdapter(value = KudosReflectiveTypeAdapterFactory::class)
data class User(
    val id: Long, 
    val name: String,
    val age: Int = -1,
    val tel: String = ""
) : KudosValidator {
    constructor() { // Generated default no-arg constructor
        super() // Call the default no-arg constructor of the parent class
        init<User>() // Call the init block of User(including the initializations of the declared properties)
        this.age = -1 // Initialize the property with the default value from the primary constructor
        this.tel = "" // Initialize the property with the default value from the primary constructor
    }

    // Function for validating null safety
    override fun validate(status: Map<String, Boolean>) {
        validateField("id", status)
        validateField("name", status)
    }
}

Use Gson to parse the JSON text:

val user = kudosGson().fromJson("""{"id": 12, "name":  "Benny Huo"}""", User::class.java)
println(user) // User(id=12, name=Benny Huo, age=-1, tel=)

If the JSON lacks the id or name field, the parsing fails, ensuring that the properties of User is null safe.

4. Collections

Properties of collection types declared in classes annotated with @Kudos, such as List or Set, will be handled carefully in the validate function to ensure the null safety for their elements. However, if the type to be parsed is like List<User>, Kudos will not be able to provide null safety guarantees at runtime because it cannot obtain whether the element type is nullable.

To solve this problem, Kudos provides KudosList and KudosSet. You can use these types in the required scenarios to ensure the null safety of elements, for example:

val list = kudosGson().fromJson("""[null]""", typeOf<KudosList<User>>().javaType)
// java.lang.NullPointerException: Element cannot be null for com.kanyun.kudos.collections.KudosList.

For more test cases, see kudos-compiler/testData.

Benchmarks

Kudos is a little slower than the according JSON tool it supports for nullability check it provides.

The code of nullability check has been optimized carefully which is generated by Kotlin Compiler directly without any runtime reflection or annotation stuffs. Based on our tests, Kudos.Gson costs only 10%~20% more time than pure Gson when deserializing in exchange for null safety.

We also find out that Kudos.Gson performs better than Moshi on initializing. So it should be a better choice for low frequency deserialization scenarios with both performance(comparing to Moshi) and null safety(comparing to Gson).

TIPS: We are working on a benchmarks project on JSON tools and will bring it out later on.

Version Support

Kotlin

Since the ABI of Kotlin compiler plugins has not been stable yet, there may be some compatibility issues between different versions. We strongly recommend using the Kudos version corresponding to the Kotlin version, for example, Kotlin 1.8.20 with Kudos 1.8.20-x.y.z.

BTW, We also provide full support for the experimental K2 compiler.

Gson

Kudos is tested on Gson 2.4-2.10. We will also follow up with the new versions of Gson in the future.

Jackson

Kudos is tested on Jackson 2.12.0-2.15.0. We will also follow up with the new versions of Jackson in the future.

Feedback

If you encounter any problems, please feel free to submit an issue.

License

Copyright (C) 2023 Kanyun, Inc.

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

   http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.