Reaktive binary compatibility: how we achieve it

Yury
Bumble Tech

--

As part of the Bumble family — the parent company operating Badoo and Bumble apps — one of my main projects was involved in the team that created Reaktive library — reactive extensions on pure Kotlin.

Whenever possible, any library must maintain binary compatibility. Whenever various versions of a library are not compatible in terms of their dependencies, there are going to be crashes in runtime. We may encounter this problem when adding Reaktive support to MVICore.

In this article I am going to briefly explain you what binary compatibility: its peculiarities in the case of Kotlin; how it has been supported at JetBrains, and how it is now supported at Bumble as well.

The problem of binary compatibility in Kotlin

Let’s say we have a wonderful library. com.sample:lib:1.0, with the following class:

Based on this library we have created a second library, com.sample:lib-extensions:1.0. Among its dependencies is com.sample:lib:1.0. For example, it contains the factory method for A class:

Then we release a new version of our library com.sample:lib:2.0 with the following modification:

This modification is entirely compatible from the point of view of Kotlin, right? With the default parameter we can continue to use the construct val a = A(a), but only if all the dependencies are fully re-compiled. Default parameters are not part of JVM and are implemented using a special synthetic constructor of class A, which contains all the class fields in its parameters. If we get dependencies from the Maven repository, they are already assembled and we are unable to re-compile them.

When a new version, com.sample:lib, comes out, we add it to our project straightaway. After all, we want to be up-to-date! New functions, new fixes, new bugs!

In this case we get a crash in runtime. The createA function in the bytecode attempts to call the constructor of class А with one parameter, but such constructor does not exist in the bytecode. Of all the dependencies with the same group and name, Gradle will choose the one which has the most recent version and will add it to the build.

Most probably, you have already come across binary incompatibility in your projects. I personally encountered this when I migrating our applications to AndroidX.

You will find more about binary compatibility in the following articles: Evolving Java-based APIs 2 from the creators of Eclipse; and in a recent article Public API challenges in Kotlin by Jake Wharton.

Ways of achieving binary compatibility

You might think that all you need to do is to introduce changes that are compatible, for example, add constructors with a default value when adding new fields, or add new parameters to the functions by overloading the method with a new parameter, etc. But when doing either of these it is too easy to make a mistake so various tools have been created for checking whether two different versions of a library are binary-compatible, including:

  1. Java API Compliance Checker
  2. Clirr
  3. Revapi
  4. Japicmp
  5. Japitools
  6. Jour
  7. Japi-checker
  8. SigTest

They accept two JAR files and return a result indicating the extent to which they are compatible.

However, we are developing a Kotlin library which so far, it only makes sense to use from Kotlin. And this means that 100% compatibility will not always be necessary — for example, for internal classes. Despite being public in bytecode, it’s very unlikely they will be used outside Kotlin code. For this reason, in order to retain binary compatibility, JetBrains uses Binary compatibility validator for kotlin-stdlib. The main principle is that a dump of all the public API is created from the JAR file and written to a file. This file forms the baseline for all further checks, and it looks like this:

Once changes have been made to the library’s source code, the baseline is generated again and compared to the current baseline — and, should there have been any changes to the baseline, the test will terminate with an error. These changes can be rewritten, using -Doverwrite.output=true. An error results even if the changes that have occurred are binary-compatible. This is necessary to ensure the baseline gets updated on time, and that its changes get seen directly in the pull request.

Binary compatibility validator

Let’s explore how this tool works. Binary compatibility is achieved at JVM (bytecode) level and is not dependent on language. It is entirely possible to replace the implementation of the Java class with Kotlin, without compromising binary compatibility (and vice versa).

First, we need to know which classes the library contains. Even in the case of global functions and constants, a class is created with a file name and suffix Kt, for example, ContinuationKt. In order to get all the classes, we use the class JarFile from JDK, obtain pointers to each class and pass them to org.objectweb.asm.tree.ClassNode. This class allows us to determine the visibility of the class, its methods, fields and annotations.

During compilation Kotlin adds its runtime @Metadata annotation to each class, so that kotlin-reflect can restore Kotlin representation of the class before it gets transformed into bytecode. This is what it looks like:

You can obtain @Metadata annotation from ClassNode and convert it into KotlinClassHeader. This has to be done manually since kotlin-reflect does not work with ObjectWeb ASM.

Metadata is needed to correctly handle internal, since this does not exist in the bytecode. Changes to internal classes and functions cannot influence the users of the libraries, even though they are a public API in the bytecode.

From Metadata you can find out about the companion object. Even if we declare it private, it is still stored in the public static field, Companion, and that means that this field is subject to the requirement for binary compatibility.

Of the annotations necessary it is also worth drawing attention to @PublishedApi for classes and methods which are used in public inline functions. The body of these functions remains where they are called, and that means that the classes and methods in them must be binary-compatible. If an attempt is made to use non-public classes and methods in these functions, the Kotlin compiler will return an error and will offer to tag them with a @PublishedApi annotation.

The class inheritance tree and interface implementation are important for supporting binary compatibility. We cannot, for example, simply remove a given interface from the class, whereas it is quite easy to obtain a parent class and interfaces.

Object has been removed from the list, as there is absolutely no point tracking it.

Inside the validator there are a lot of different additional checks specific to Kotlin: verification of defaults methods in the interfaces via Interface$DefaultImpls; ignoring $WhenMappings classes for the work of the when operator; and others.

Then it’s necessary to go through all the ClassNodes and obtain their MethodNodes and FieldNodes. From the class signatures, and their fields and methods we obtain ClassBinarySignature, FieldBinarySignature and MethodBinarySignature, which are declared locally in the project. They implement the MemberBinarySignature interface, are able to determine their public visibility using the isEffectivelyPublic method, and provide their signature in a readable val signature: String format.

After the ClassBinarySignature list is obtained, this can be written to a file or to the memory using the dump(to: Appendable) method, and may be compared with the baseline, which is what happens in the RuntimePublicAPITest test:

Having committed a new baseline, we obtain changes in a readable format, such as, for example, in this commit:

Using a validator in your project

It is extremely simple to use. Copy binary-compatibility-validator to your project and change its build.gradle and RuntimePublicAPITest:

In our example one of the test functions of the RuntimePublicAPITest file looks like this:

Now we run ./gradlew :tools:binary-compatibility:test -Pbinary-compatibility-override=false for every pull request and make developers update baseline files on time.

Fly in the ointment

However, this approach also has its downsides.

Firstly, we ourselves have to analyse changes to the baseline files. Such changes do not always lead to binary incompatibility. For example, implementing a new interface leads to the following difference in the baseline:

Secondly, it uses tools that were never intended for this purpose. Tests should not have side-effects such that of some file gets written to disk, which, resulting in it being used by this test, nor should they ever pass it parameters via environment variables. It would be great to use this instrument as a Gradle task. But we really don’t want to change something in the validator ourselves, so making it easy for all its changes to be pulled from the Kotlin repository, because in future there might be new constructs in the language which would need to be supported.

And thirdly, only JVM is supported.

Conclusion

Binary compatibility and a time response to a change in its state can be achieved with the help of a Binary compatibility validator. Its use in this project required changing just two files and adding tests to our CI. Despite this solution having several drawback it is still quite convenient to use. Reaktive is now going to try to support binary compatibility for JVM in the same way that JetBrains does for the Kotlin Standard Library.

Thanks for reading!

Update: Last weekend Kotlin team finally released Binary compatibility validator as standalone Gradle Plugin. It uses the same principles I described in this article and now can be used much easier!

--

--