1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
|
# Kotlin Public API binary compatibility validation tool
This tool allows to dump binary API of a Kotlin library that is public in sense of Kotlin visibilities
and ensure that the public binary API wasn't changed in a way that make this change binary incompatible.
## How to run
Compile and run tests. `CasesPublicAPITest` verifies the tool itself,
and `RuntimePublicAPITest` dumps the public API of `kotlin-stdlib`,
`kotlin-stdlib-jdk7/8`, and `kotlin-reflect` jars,
which must be built beforehand with gradle. Use `clean assemble` tasks,
since the incremental compilation currently doesn't produce all the required output.
When substantial changes are made to the public API, it may be convenient to overwrite
the entire dump and compare changes later before committing: pass `-Doverwrite.output=true`
property to the test to do so.
Also you can use shared run configuration "Binary compatibility tests", which also
overwrites the results when they differ.
## What constitutes the public API
### Classes
A class is considered to be effectively public if all of the following conditions are met:
- it has public or protected JVM access (`ACC_PUBLIC` or `ACC_PROTECTED`)
- it has one of the following visibilities in Kotlin:
- no visibility (means no Kotlin declaration corresponds to this compiled class)
- *public*
- *protected*
- *internal*, only in case if the class is annotated with `InlineExposed`
- it isn't a local class
- it isn't a synthetic class with mappings for `when` tableswitches (`$WhenMappings`)
- it contains at least one effectively public member, in case if the class corresponds
to a kotlin *file* with top-level members or a *multifile facade*
- in case if the class is a member in another class, it is contained in the *effectively public* class
- in case if the class is a protected member in another class, it is contained in the *non-final* class
### Members
A member of the class (i.e. a field or a method) is considered to be effectively public
if all of the following conditions are met:
- it has public or protected JVM access (`ACC_PUBLIC` or `ACC_PROTECTED`)
- it has one of the following visibilities in Kotlin:
- no visibility (means no Kotlin declaration corresponds to this class member)
- *public*
- *protected*
- *internal*, only in case if the class is annotated with `InlineExposed`
> Note that Kotlin visibility of a field exposed by `lateinit` property is the visibility of it's setter.
- in case if the member is protected, it is contained in *non-final* class
- it isn't a synthetic access method for a private field
## What makes an incompatible change to the public binary API
### Class changes
For a class a binary incompatible change is:
- changing the full class name (including package and containing classes)
- changing the superclass, so that the class no longer has the previous superclass in
the inheritance chain
- changing the set of implemented interfaces so that the class
no longer implements interfaces it had implemented before
- changing one of the following access flags:
- `ACC_PUBLIC`, `ACC_PROTECTED`, `ACC_PRIVATE` — lessening the class visibility
- `ACC_FINAL` — making non-final class final
- `ACC_ABSTRACT` — making non-abstract class abstract
- `ACC_INTERFACE` — changing class to interface and vice versa
- `ACC_ANNOTATION` — changing annotation to interface and vice versa
### Class member changes
For a class member a binary incompatible change is:
- changing its name
- changing its descriptor (erased return type and parameter types for methods);
this includes changing field to method and vice versa
- changing one of the following access flags:
- `ACC_PUBLIC`, `ACC_PROTECTED`, `ACC_PRIVATE` — lessening the member visibility
- `ACC_FINAL` — making non-final field or method final
- `ACC_ABSTRACT` — making non-abstract method abstract
- `ACC_STATIC` — changing instance member to static and vice versa
|