# Mojaloop Versioning, A Proposal
Note: This document is a living draft of a proposal for versioning within Mojaloop. Once the proposal is ready, it will be submitted to the CCB for approval.
The aim is to produce a proposal that keeps the versioning Scheme simple to use and clear regarding compatibility issues. However, it needs to include all the details needed for a Mojaloop ecosystem as well.
Goal: Propose a standard for a new 'Mojaloop Version', which embodies:
- Helm: Individual Service Versions, Monitoring Component Versions
- API Versions: FSPIOP API, Hub Operations / Admin API, Settlement API
- Internal Schema Versions: Database Schema and Internal Messaging Versions
# Versioning Strategies/Background (Literature Review)
How do current systems handle versioning? Give a brief overview of the current space.
- Most best practices follow semantic versioning for APIs, this will be covered more in #1198 (opens new window)
# Demonstrates zero-downtime deployment approaches with Kubernetes 
- in order to support rollbacks, the services must be both forward and backwards compatible. consecutive app versions must be schema compatible
- 'Never deploy any breaking schema changes', separate into multiple deployments instead
For example, start with a PERSON table:
PK ID NAME ADDRESS_LINE_1 ADDRESS_LINE_2 ZIPCODE COUNTRY
And we want to break this down (normalize) into 2 tables, PERSON and ADDRESS:
#person PK ID NAME #address PK ID FK PERSON_ID ADDRESS_LINE_1 ADDRESS_LINE_2 ZIPCODE COUNTRY
If this change were made in one migration, 2 different versions of our application won't be compatible. Instead, the schema changes must be broken down:
- Create ADDRESS table
- App use the PERSON table data as previously
- Trigger a copy of data to the ADDRESS table
- The ADDRESS now becomes the 'source of truth'
- App now uses the ADDRESS table data
- Trigger a copy of new added to address to the PERSON table
- Stop copying data
- Remove extra columns from PERSON table
This means for any one change of the database schema, multiple application versions will need to be created, and multiple deployments must be made in succession for this change to be made.
-  also notes how simple Kubernetes makes deploying such a change * rolling upgrade deployments * Tip: make sure your health endpoint waits for the migrations to finish!
- Q: so how do we make big changes that touch both the database schema and the API? * this seems really hard, and would need a lot of coordination * If we don't design it correctly, it could mean that a single schema change could require all DFSPs to be on board * This is why I think the API version and Service version should be unrelated. We should be able to deploy a new version of a service (which runs a migration), and supports an old API version
# Using a schema registry for Kafka Messages 
-  suggests some approaches, such as using a schema registry for kafka messages, such as Apache Arvo (opens new window)
- This adds a certain level of 'strictness' to the messages we produce, and will help enforce versioning
- Adds a separate 'schema registry' component, which ensures messages conform to a given schema. This doesn't really help enforce versioning, and leaves the work up to us still, but does give more guarantees about the message formats.
# Backwards and Forwards Compatibility , 
- "The Robustness principle states that you should be “liberal in what you accept and conservative in what you send ”. In terms of APIs this implies a certain tolerance in consuming services." 
- Backwards Compatibility vs Backwards Incompatibility :
- Generally, additions are considered backwards compatible
- Removing or changing names is backwards incompatible
- It's more something to assess on a case-by-case basis, but Google's API Design Doc (opens new window) helps lay out the cases.
# Mojaloop Ecosystem
When discussing versioning we need to be clear that we are versioning interfaces for various parties.
The following section will outline the versioning proposal.
# A “Mojaloop Version”
A Mojaloop Version x.y.z can be defined that can encompass the versions of all the three APIs included (detailed below). In the version x.y.z, ‘x’ indicates the Major version and ‘y’ a minor version, similar to the Mojaloop FSPIOP API versioning standards; ‘z’ represents the ‘hotfix’ version or a version released with the same major, minor version x.y but to keep things simple, there is a need to bundle all the components included in the Mojaloop ecosystem indicating what all items are included there.
In effect we may say Mojaloop version x.y includes
- Mojaloop FSPIOP API
- Maintained by the CCB (Change Control Board)
- Uses x.y format
- Currently version v1.0, v1.1 and v2.0 are in the pipeline
- Settlement API
- Maintained by the CCB
- To use x.y format
- Currently version v1.1 and v2.0 is in the pipeline
- Admin / Operations API
- Maintained by the CCB
- To use x.y format
- Can use version v1.0
- Maintained by the Design Authority
- Uses x.y.z format
- PI (Program Increment) + Sprint based versioning.
Note: PI + Sprint based versioning make sense in the context of the current Mojaloop Program Increments, but will need to be revised at a later date.
- Bundles compatible versions of individual services together
- Internal Schemas
- Maintained by the Design Authority
- DB Schema x.y
- Internal messaging Schema (Kafka) x.y
|- FSPIOP API||CCB||x.y||Major.Minor|
|- Settlement API||CCB||x.y||Major.Minor|
|- Admin/Operations API||CCB||x.y||Major.Minor|
|- DB Schema||Design Authority||x.y||Major.Minor|
|- Internal Messaging||Design Authority||x.y||Major.Minor|
For example: Mojaloop 1.0 includes
- FSPIOP API v1.0
- Settlements API v1.1
- Admin API v1.0
- Helm v9.1.0
- Individual services' versions
- Monitoring components versions
- Internal Schemas
- DB Schema v1.0
- Internal messaging version v1.0
|- FSPIOP API||CCB||1.0|
|- Settlement API||CCB||1.1|
|- Admin/Operations API||CCB||1.0|
|- DB Schema||Design Authority||1.0|
|- Internal Messaging||Design Authority||1.0|
- The advantage of this strategy is primarily simplicity. A given version say - Mojaloop v1.0 can just be used in discussions which then refers to specific versions of the three APIs - FSPIOP, Settlements, Admin APIs, along with the Helm version that is a bundle of the individual services which are compatible with each other and can be deployed together. Along with these, the Schema versions for the DB and Internal messaging to communicate whether any changes have been made to these or not since the previously released version.
- The other advantage, obviously, is that it caters for everyone who may be interested in differing levels of details , whether high level or detailed. Because of the nature of the major and minor versions, it should be easy for Users and adopters to understand compatibility issues as well.
As described in section 3.3 of the API Definition v1.0 (opens new window), whether or not a version is backwards compatible is indicated by the Major version. All versions with the same major version must be compatible while those having different major versions, will most likely not be compatible.
Important Note: Hub operators will likely need to support multiple versions of the FSPIOP API at the same time, in order to cater for different participants as they can't all be expected to upgrade at the same time.
# Breaking down the “Mojaloop Version”
This section aims to break down the above proposed mojaloop version into its constituent parts, and provide support for the above proposed versioning strategy
The Mojaloop Spec (opens new window) already outlines many of the decisions made around versioning APIs.
In terms of common best practices, there are many approaches for requesting different versions, including adding in a version in the url, but let's not worry about this because the spec already lays this out for us, using the HTML vendor extension: 184.108.40.206 Http Accept Header (opens new window)
As for version negotiation, the spec also states that in the event of an unsupported version being requested by a client, a HTTP status 406 can be returned, along with an error message which describes the supported versions. 220.127.116.11 Non-Acceptable Version Requested by Client (opens new window)
Another best practice around versioning is specifying to what level clients may request specific apis.
- In a development environment, many APIs will allow specificy up to the BUGFIX version, i.e. vX.X.X
- In production however, this is limited to Major versions only, e.g. v1, v2
- e.g. The Google API Platform supports only major versions, not minor and patch versions
- Given the new features that may become available with v1.1 of the Mojaloop API, we might want to allow participants to specify MAJOR and MINOR versions, i.e. vX.X. This practice should be avoided however, since minor versions should be backwards compatible
Participants using the same MAJOR version of the API should be able to interact. Participants on different MAJOR versions are not able to interact. For example, a participant on API v1.1 can send transfers to another participant on v1.0, but not to a different participant on v2.0.
This section deals with how Mojaloop services interact within a given deployment. Here, we attempt to propose questions such as "should an instance of central-ledger:v10.0.1 be able to talk to ml-api-adapter:v10.1.0? How about ml-api-adapter:v11.0.0?"? or "how do we make sure both central-ledger:v10.0.1 and central-ledger:v10.1.0 talk to the database at the same time?"
There are two places where this happens:
- Where services interact with saved state - MySQL Percona Databases
- Where services interact with each other - Apache Kakfa and (some) internal APIs
This implies we need to version:
- the database schema
- messages within Apache kafka
- need to make sure the right services can appropriately read the right messages. E.g. Can mojaloop/ml-api-adapter:v10 .1.0 publish messages to kafka that mojaloop/central-ledger:v10.0.1 can understand?
- Q: If we decide to make breaking changes to the message format, how can we ensure that messages in the kafka streams don't get picked up by the wong services?
# Internal Schemas
todo: anything to be said here?
Currently, we use the lime protocol for our kafka message formats: https://limeprotocol.org/
Also refer to the mojaloop/central-services-stream readme for more information about the message format.
The lime protocol provides for a type, field, which supports MIME type declarations. So we could potentially handle messages in a manner similar to the API above (e.g. application/vnd.specific+json). Versioning messages in this manner means that consumers reading these messages would need to be backwards and fowards compatible (consecutive message versions must be schema compatible).
- Q. does it make sense to put the version in the Kafka topic?
- One example, ml-api-adapter publishes messages to the prepare topic
- If we add versioning to this, ml-api-adapter:v10.0.0 publishes messages to a prepare_v10.0 topic, and a new instance of the ml-api-adapter:v10.1.0 will publish to the prepare_v10.1 topic.
- subscribers can subscribe to whichever prepare topic they want, or both, depending on their own tolerance to such messages
- This may have some serious performance side effects
- Another potential option would be to allow for a message 'adapter' in the deployment. Say the ml-api-adapter:v10.1.0 is producing messages to a prepare_v10.1 topic, and there is no corresponding central-ledger in the deployment to read such messages, we could have an adapter, which subscribes to prepare_v10.1, reformats them to be backwards compatible, and publishes them to prepare_v10.0 in the old format.
Such an approach would allow for incremental schema changes to the messaging format as services are gradually upgraded.
All in all, I didn't see too much about this subject, so we'll likely need to return later down the line.
# Version Negotiation
todo: @sam Discuss how to deal with version negotiation strategy
# Long Term Support
todo: Discuss how long term support fits into the versioning proposal. I don’t think we want to get into too much detail, but more outline what it might look like
Mention current (lack of) lts support, current PI cadence
# Appendix A: Definitions
- service: Mojaloop follows a microservices oriented approach, where a large application is broken down into smaller micro services. In this instance, Service refers to a containerized application running as part of a Mojaloop deployment. At the moment, this takes the form of a Docker container running inside of a Kubernetes cluster. e.g. mojaloop/central-ledger is the central-ledger service
- service version: The version of the given service. This currently doesn't follow semantic versioning, but may in the future e.g. mojaloop/central-ledger:v10.0.1. The current approach is described in more detail in the standards /Versioning doc (opens new window).
- helm: Helm is an application package manager that runs on top of Kubernetes. It may also be referred to as the "deployment". A single helm deployment runs many different services, and MAY run multiple versions of the same service simultaneously. We also refer to the deployment as it's repo, mojaloop/helm interchangeably.
- helm version: A helm version is the version of the packaged helm charts, e.g.mojaloop/helm:v1.1.0
- interface: An interface is the protocol by which a Mojaloop switch interacts with the outside world. This includes interactions with Participants (DFSPs) who transfer funds through the switch, hub operators running a Mojaloop switch, and admins performing administrative functions.
- api: Application Programming Interface - in most cases referring to the FSPIOP-API a.k.a. Open API for FSP Interoperability defined here (opens new window).
- api version: The Version of the FSPIOP-API, e.g. FSPIOP-API v1. For the purposes of this document, it refers to the contract between a Mojaloop Switch and Participants (DFSPs) who implement the FSPIOP-API
 LTS versioning within nodejs. This is a great example of an LTS strategy, and how to clearly communicate such a strategy.  Semantic Versioning Reference  https://www.ben-morris.com/rest-apis-dont-need-a-versioning-strategy-they-need-a-change-strategy/  https://cloud.google.com/apis/design/compatibility  Nicolas Frankel - Zero-downtime deployment with Kubernetes, Spring Boot and Flyway  Stackoverflow - Kafka Topic Message Versioning