People sitting together at the same table

Last year, we started migrating our single-repo, multi-module code-base (160+ backend interfaces) from the old OSGI Framework to a new multi-repo cloud-native architecture with the following characteristics:

  • Each Interface is hosted in a separate git repository and has its application lifecycle.
  • Interfaces are delivered as a container image for an orchestration platform to a container registry. In our case, Helm is used so the helm deployment manifests are delivered to a helm registry.

A separate code repository allows for faster and more frequent releases but also introduces a big challenge of creating or updating configuration for the operation team due to the number of releases and variety of interfaces.

In this article, we would like to focus on the documentation aspect of having a cloud-native architecture and share our way of documentation with the hope that it may help other teams and also benefit from the reader’s recommendations.

Old school documentation

Before we talk about the challenges and our solution, it would be a good idea to see what we have been doing so far. We used to have various documents based on Microsoft Word and some diagrams based on the application Enterprise Architect.

We had the following documents:

  • Software Architecture: Described the overall system architecture as well as all interfaces(OSGI Bundles) specific architecture. The document was based on Microsoft Word and saved on the git repository along with the source code.
  • Operation Manual: Focused on the operational aspect of our applications and provided some introduction to installing the platform (Red Hat Fuse 6) and some of the main operations. This document was based on Microsoft Word and saved on the git repository along with the source code.
  • Release Notes: Provided the user stories implemented in the application and also the configuration updates mostly implemented by the operation team. This document was saved as a Microsoft Word file and versioned on the git repository along with the source code.
  • Confluence pages: Had all kinds of outdated information about some aspects of the applications used mostly by developers.

In addition, each of our developers had their private how-to guides.

Old school documentation delivery

Since we were working in an agile team, the documentation artifacts were tagged with the application version and saved as a PDF and delivered to the customer via email or some file-sharing platform.

As you can guess, it did bring lots of issues such as:

  • We could not do a diff-check on the code repository for binary files (pdf, docx, …)
  • The email recipients could not get the email for whatever reason and basically, some shareholders would miss the information.
  • Copying configuration would also take unknown or special characters.

Migration project

With the new migration project, we started to migrate our complete source code to new cloud-native architecture and one of the important goals of our migration was zero-to-hero documentation, which should have the following qualities:

  • Is single source-of-truth
  • Ease of versioning
  • Various versions of the same application documentation are easily accessible
  • Easy to deliver and all stakeholders can access it at any time

We expect the above quality measures can:

  • Allow faster and easier onboarding of new team members.
  • Reduce maintenance effort of the project, hence increase in the efficiency of developers.
  • Better communication with our stakeholders such as product owners, solution architects, operation teams, and interface consumers.

Changing the architecture can impose big changes on the way of working. For example concerning documentation:

  • What are the documentation artifacts?
  • Where do we keep documentation?
  • How do we deliver documentation?
  • How is it versioned?
  • How can we better review artifacts before delivering?

In the following section, we’ll talk about our solution and how we achieved the above-mentioned goals.

Our solution

The very first step we took to achieve our documentation goal was, to migrate from binary formats to text-based formats which solves the problem of reviewing and diff-checks.

We used AsciiDoc for our technical writing. The reasons for our decision were:

  • It is easy to learn.
  • It can be used by any text editor. We use Asciidoctor and Visual Studio Code.
  • Since it is text-based, it can be versioned on any version control system.
  • It can be published in multiple output formats.
  • Most importantly, it allows concentration on writing.
  • There are plenty of tools to draw diagrams as text such as plantuml.

Now that we have our documentation as code, we need to:

  • Identify the kind of documentation artifacts we have.
  • Chose tools to build our documentation artifacts.
  • Think of a delivery method for our new documentation artifacts.

In our new project we produce the following artifacts:

  • Macro-Architecture: This document gives a big picture of the project. All interfaces should follow the design/constraints specified in this document. The sections that are the same as all other interfaces will be explained in this document to avoid redundant work. If any interface requires a deviation of design, it will be documented in the interface-specific architecture document. This document is based on the arc42 template.
  • Overall-Operation-Manual: Provides guides about overall infrastructure/platform requirements and settings. It also provides a general debugging guide and references.
  • Release-Notes: Provides specific information on each interface and is produced for each release.
  • Interface-Architecture: Provides interface-specific architecture documentation. It follows the Macro-Architecture and provides additional information if required.
  • Operation-Manual: Guide how to configure the interface and describe technical properties. This document is produced for each interface separately.
  • Developer Guide: Provides all the required information on how to develop, extend and run the interface. It provides information such as how to run mock systems, configuration, and anything that helps not to rely on the original developer for the tricks to run. Any team member, that has never seen its code, should be able to at least build and run the interface by reading this document.

In addition to the above documents, there are other documents such as the functional requirements of the interface.

The Macro-Architecture and Overall-Operation-Manual are saved separately in their git repositories. The rest of the documents are interface-specific and are saved along with the source code and are versioned the same as the application version.

To solve the problem with versioning documents we commit to plain-text languages such as Asciidoc and Markdown.


Now that we have our document as a source, we can apply whatever transformation and formatting we want.

The challenge: The source code is scattered in many git repositories and we need to combine them with the least effort using some kind of engine which we call doc-centralizer (see below).

Building Documentation with doc-centralizer

We do not need to reinvent the wheel. It already exists and is perfectly documented!

Antora is a multi-repository documentation site generator for tech writers who are writing text or documentation in AsciiDoc. The configuration of Antora for generating the final output is done in a declarative manifest called antora-playbook.yml. In this playbook, we enter the source code repository URL and branches that should be included in the documentation.

Our documentation is built based on various components such as interface-architecture, interface operation manual etc. Each of these components could have various versions. For example, each release branch can have its own version.

To understand better, we can look into the documentation of Camel framework.

Camel Framework Documentation

In the above example, we can click and switch to the following components:

  • User Manual
  • Camel Components
  • Camel K
  • Camel Kafka Connector

Also, we can choose a specific version of a component. For example, we can click on version 2.x under Camel components to see the documentation. This is especially useful for readers to fast-switch and compare various versions.

Note: discussing Antora is not the focus of this article and the usage of this framework can be best learned on its website.


We use Anotra in a pipeline to fetch all the documentation and combine them into a static website. Once we have our website, we can serve it using any webserver anywhere!

We chose to deliver our documentation as a container image to a container registry which can be run easily on any container platform such as Kubernetes or Openshift.

The build and delivery process is automated and uses something like the below dockerfile to build a container image:

FROM as builder
RUN npm i asciidoctor.js asciidoctor-plantuml
COPY . /antora/
RUN antora generate --stacktrace antora-playbook.yml
COPY --from=builder /antora/doc-output/ /var/www/html/r

The provided dockerfile is just one simple example of how to build the container. Depending on the requirements, we can also use other webservers such as Nginx or enable authentication and authorization which are not the focus of this article.

Our Recommendation:

  • Team culture is an essential factor to create useful and productive documentation. Build a team culture to commit to detailed documentation and enforce it on the Definition-Of-Done list or any review processes.
  • Use text-based documentation. It allows focusing on writing and documentation. The learning curve might be slow at first but it solves many problems in the long run.
  • Version documentation along the source code.
  • Less confluence (or similar) pages.
Picture Reza  Nirumand

Author Reza Nirumand

Reza Nirumand is a Senior Software Engineer at adesso located in Dortmund. He is involved in various projects in the Banking industry and he is responsible for the designing and delivering of cloud-native applications.

Save this page. Remove this page.