Versioning policy
- Single version number
- Library versioning policy
- Application versioning impact
- Changing Merkle serialization
- Changes to logs
- Changes to messages
- Modifying gossip
- Changelog
Kolme is a Rust library, and as such can follow a standard semver-inspired versioning scheme for backwards compatibility of the API. However, due to the nature of how Kolme is used, the story is a bit more complicated. In particular:
- Changes to Kolme may impact downstream applications using its APIs. This is the standard versioning issues libraries face.
- New versions of Kolme may modify any number of protocol components exposed to the outside world, network APIs and serialization formats being the most obvious.
- When applications themselves make changes, they may change their own APIs, serialization, or application logic, all of which would make it impossible to reproducibly reexecute prior transactions.
The last point is mostly out of scope for this document. It is a responsibility of application authors, and is enabled by the version upgrade system provided by Kolme. Please see that document for a better understanding of the goals in this document.
The purpose of this document is to ensure:
- Developers of Kolme make changes in a way that allows for backwards compatibility with old serialized data.
- We have a clear signposting mechanism for communicating breaking changes that downstream application developers need to handle.
Single version number
In theory, we could use multiple version numbers for Kolme:
- Version each sublibrary (like merkle-mapseparately).
- Version the serialized block format.
- Version the network API.
And we may ultimately decide to go in that direction. However, we're early in Kolme's development, and that level of complexity isn't currently warranted. Instead, we currently simply track one version number of Kolme: the version of the kolme crate itself. This represents all different pieces of the system as one.
We follow Semantic Versioning (SemVer) for this version number.
Library versioning policy
As a Rust library, Kolme does not need to reinvent any wheels. We can follow standard Rust versioning rules. These are documented at length.
However, at its current state of development, Kolme does not strive to keep stable APIs. It is primarily an internal FP Block tool used for our internal development. As such, we strive to reduce unnecessary code breakage, but need not insist on such compatibility.
This will change at some point in the future, but not yet.
As a result of this, we currently have no specific policy around whether a change below results in a major, minor, or patch version bump. We'll refine this over time.
Application versioning impact
Applications maintain a version string to indicate compatibility with old block production, as discussed in the version upgrading guide. As a simple, conservative measure: any time you release any new version of the application, you should bump the application version number (code/chain version) and go through the full version upgrade process.
Technically, however, you only need to perform such a version bump if a change could result in differences in block production. In practice, almost any change could result in that, even a simple bump to a decimal library (since it may result in slightly different arithmetic results).
Unless explicitly stated otherwise, any change discussed below should be considered as requiring a new application version.
Changing Merkle serialization
Merkle serialization is the most important piece of Kolme to maintain compatibility for. Without this, old block data will be unreadable by newer versions of the library.
Here are some basic rules:
- Any data structures that may be modified in the future should provide MerkleSerialize/MerkleDeserializeimpls, instead of theirRawvariants.
- Any modification to the serialized data must result in a bump to the merkle_versionmethod's return value.
- As a strong recommendation, new fields should be added at the end of a data structure.
- Any newly added fields can be serialized as normal, but when deserializing, you need to check the version number and ensure the field is parsed only for versions it was serialized in.
- New fields must include a fallback value for parsing old data. This could either be via wrapping with Option, or providing a default value.
If all that seems a bit abstract, the easiest way to understand it is via the merkle-map versioning test code, e.g.:
#![allow(unused)] fn main() { #[derive(Clone, PartialEq, Eq, Debug)] struct Person0 { name: String, age: u16, } impl MerkleSerialize for Person0 { fn merkle_serialize(&self, serializer: &mut MerkleSerializer) -> Result<(), MerkleSerialError> { let Self { name, age } = self; serializer.store(name)?; serializer.store(age)?; Ok(()) } fn merkle_version() -> usize { 0 } } impl MerkleDeserialize for Person0 { fn merkle_deserialize( deserializer: &mut merkle_map::MerkleDeserializer, version: usize, ) -> Result<Self, MerkleSerialError> { Ok(Self { name: deserializer.load()?, age: deserializer.load()?, }) } } #[derive(Clone, PartialEq, Eq, Debug)] struct Person1 { name: String, age: u16, street: String, } const DEFAULT_STREET: &str = "Default street"; impl From<Person0> for Person1 { fn from(Person0 { name, age }: Person0) -> Self { Self { name, age, street: DEFAULT_STREET.to_owned(), } } } impl MerkleSerialize for Person1 { fn merkle_serialize(&self, serializer: &mut MerkleSerializer) -> Result<(), MerkleSerialError> { let Self { name, age, street } = self; serializer.store(name)?; serializer.store(age)?; serializer.store(street)?; Ok(()) } fn merkle_version() -> usize { 1 } } impl MerkleDeserialize for Person1 { fn merkle_deserialize( deserializer: &mut merkle_map::MerkleDeserializer, version: usize, ) -> Result<Self, MerkleSerialError> { Ok(Self { name: deserializer.load()?, age: deserializer.load()?, street: if version == 0 { DEFAULT_STREET.to_owned() } else { deserializer.load()? }, }) } } }
Changes to logs
Changing log messages may seem like something that doesn't affect downstream. However, it's something that can cause breakage in two ways:
- Some log messages may be relied upon and parsed by downstream tools.
- Since hashes of logs are stored in blocks, any change in logging will impact reproducibility of blocks.
Make sure that any change to logs is well documented in the changelog.
Changes to messages
This is more obvious than logs. Any change to built-in messages (admin, fund transfer, etc.) will result in changes to transactions and therefore blocks. This doesn't just apply to the API itself, but any change in the handling may result in differences in binary output.
Modifying gossip
Gossip modifications are less severe than the changes above. They impact the network protocol, but do not directly affect block production. Keeping compatibility with the immediately prior version of gossip is a good thing for seamless migrations.
Changelog
The changelog for Kolme is maintained in CHANGELOG.md at the repository root, following the Keep a Changelog format.
- The initial version of the changelog is generated using git-cliff, which parses the commit history and creates a structured changelog.
- After the initial generation, the changelog is updated manually by the team for each release or significant change.
- There is no required commit message convention; all changelog updates are made directly in CHANGELOG.mdas part of the release process.
- When making changes, update the relevant sections ("Added", "Changed", "Fixed", etc.) in CHANGELOG.mdto reflect what has been done since the last release.
- The changelog is committed to the repository and should be kept up to date as part of the release process.
We're going to follow the "bump right before" strategy of bumping version numbers in Cargo.toml files just before cutting a release. That means that the repo will always have the newest released version number in the Cargo.toml files.