Deploying Docs-as-Code on AWS: Building Dynamic Documentation Sites in MkDocs and Docusaurus

Data Python Cloud Computing AWS

In this article, I will guide you step by step to create a dynamic documentation site, adaptable to any project, where you can connect your documentation to a database to extract and display data, ensuring the information is always up to date. We will also explore how to automate the entire process, from content generation to deployment in the cloud with AWS.

Romina Mendez https://example.com/norajones
2025-01-10

The solution includes support for charts and diagrams, continuous integration (CI/CD) using a simple workflow in GitHub Actions, and automatic deployment using Terraform. Let’s get started!


What is Documentation as Code?

Documentation and its updates are an important process in many companies that develop software, often carried out using different tools, many of which are paid solutions.
Therefore, in recent times, the concept of “doc as code” has emerged. This means using the same tools and workflows used in software development to manage, version, and deploy documentation.
This approach not only allows for better tracking of the documentation but also facilitates its maintenance and ensures alignment with the same best practices used in software development, not just in the code but also in the documentation .


Tools for Documentation as Code

For the development of these sites, it is essential to understand some practices and tools that allow us to implement this approach. Below is a detailed list of the most important aspects to cover in this tutorial.


Advantages of Docs-as-Code


📄 MkDocs

MkDocs is a static site generator written in 🐍Python, designed specifically for documenting projects. Its goal is to simplify creating documentation using Markdown files, which are easy to write and read.

With minimal configuration, MkDocs converts Markdown files into a navigable and well-structured documentation website, making it ideal for developers and teams who want to keep their documentation up to date.


✏️ MkDocs Material

MkDocs Material is an advanced theme for MkDocs that follows Google’s Material Design guidelines.

🚀 Key features include:


✏️ Mermaid

Mermaid is a **JavaScript** library for creating diagrams and charts from text. By integrating with MkDocs Material, Mermaid allows you to generate visualizations such as flowcharts, entity-relationship diagrams, and other charts within the documentation without external tools.


🧩 Dynamic Page: Jinja

Jinja is a library that allows embedding variables and data from Python dictionaries into HTML, making web pages dynamic. This library is commonly used for generating dynamic HTML and sending personalized emails.


🦖 Docusaurus Overview

Docusaurus is an open-source project developed by Meta in 2007 that simplifies the creation, deployment, and maintenance of documentation websites in a fast and efficient way. It allows the use of Markdown and MDX to write content, while its core built on React enables full customization of the styles to fit the specific needs of the project.

Additionally, Docusaurus supports Mermaid through the @docusaurus/theme-mermaid plugin, enabling the inclusion of charts and diagrams directly within the documentation.


🎨 Diagram as Code

Diagram as Code is an approach that allows you to create diagrams through code, rather than using traditional graphic tools. Instead of manually building diagrams, you write code in a text file to define the structure, components, and connections of your diagrams.

This code is then translated into graphical images, making it easier to integrate and document in software projects. It’s especially useful for creating and updating architectural and flow diagrams programmatically.


🎨 Diagram as Code: Example of Creating Cloud Diagrams

As previously mentioned, Diagrams allows you to generate blueprints using the icons of major cloud technologies. The representation of these diagrams is done through nodes, and in our example, we’ll use all cloud-related nodes and AWS services.

For more details on how I created this, you can read my article about Diagram as Code, and the full implementation can be found in this repository:


📚 Use Case: Creating a Documentation Site for a Machine Learning Project

In this use case, I will create a documentation site for a machine learning project involving 🏥 hospital data. The goal is to build an interactive documentation site using MkDocs initially and later migrate it to Docusaurus. The site will include both static and dynamic components to meet specific requirements, such as embedding visual diagrams and updating data dynamically from a SQLite database.


🚀 Key Features of the Documentation Site

  1. Visual Representations: I will embed diagrams created with Diagrams (Diagram as Code) to illustrate the architecture of the machine learning pipeline effectively.
  2. Dynamic Data Updates: The documentation will display the version and last update date dynamically, pulling this information from a SQLite database to ensure accuracy and relevance.
  3. Sample of data: The documentation will include a sample from the Synthea patient table, showcasing synthetic data as an example.

📄 Pages of the Site

For this reason our documentation site will have the following pages:


MkDocs Implementation

In this section, we’ll walk through the steps to set up a documentation project using MkDocs from scratch and explain its organized directory structure.

🔧 Prerequisites for MkDocs

To get started, you’ll need to install the following 🐍Python libraries:

Install MkDocs and the Material bash pip install mkdocs mkdocs-material

Install additional libraries to enable dynamic content updating bash pip install aiosql pandas sqlite3 jinja2 shutil

🔧 Mkdocs: Project Setting Up

Initialize the Project
Start by creating a new MkDocs project. Run the following commands in your terminal:
bash mkdocs new mkdocs cd mkdocs This command creates a basic MkDocs project with a default structure .

Explore the Directory Structure
Once the MkDocs site is created, you need to add the following files and folders, as they are not included by default.   Remember, the links to the repository are provided at the end of this post for your reference, and each component will be explained in detail below.

📁 docs/
  ├── 📁 img/
  ├── `architecture.md`
  ├── `glossary.md`
  ├── `index.md`
  ├── `tables.md`
  ├── 📁 template/
  │   ├── 📁 db/
  │   │   ├── 📁 data/
  │   │   │   ├── hospital.db
  │   │   ├── 📁 queries/
  │   ├── `architecture.md`
  │   ├── `glossary.md`
  │   ├── `index.md`
  │   ├── `tables.md`
  │   └── `update.py`
📁 infraestructure/
📁 github/
  ├── 📁 workflows/
  │   ├── main.yml
📄 mkdocs.yml

📂Mkdocs: Component Overview

Component Directory Description
Database (db) db Contains the SQLite database (hospital.db) and queries (metadata.sql, person.sql) to manage dynamic data. Learn more about managing SQL queries in Python in my previous article: Python Projects with SQL.
🖋️ Templates & Pages template Markdown templates: index.md, tables.md, architecture.md, glossary.md. Supports Mermaid diagrams, embedded images, and database-driven content.
🖼️ Static Content (docs) docs Final site generated by update.py, including images (img/) and dynamic content populated from template.
🌐 Infrastructure (infraestructure) infraestructure Terraform scripts (main.tf, variables.tf) to deploy an S3 bucket for documentation hosting.

📄 Mkdocs

Configuring mkdocs.yml Once we have our project structure set up, we will configure it step by step, starting with the mkdocs.yml file. This file defines the structure and settings for your documentation site. Here’s how it should be structured:

mkdocs.yml

site_name: Hospital Documentation  
nav:
  - Home: index.md
  - Synthea Tables: tables.md
  - AWS Architectur: architecture.md
  - Glossary: glossary.md
markdown_extensions:
  - pymdownx.superfences:
      custom_fences:
        - name: mermaid
          class: mermaid
theme:
  name: material                

In this configuration file, you can primarily see in the nav section the pages that will be accessible from the menu. Then, we specify the Mermaid extension, which will be explained in the next section. Finally, the theme section applies the Material theme, enabling styling and components available within this library.


✏️ Mkdocs: Mermaid Extension

As mentioned earlier, Mermaid is a JavaScript library for creating diagrams and charts from text. Below, we will see some examples. In our case, we will use it to generate an Entity Relationship Diagram (ERD) on the tables page of the documentation.

Image description

In the repository, you will be able to see how to construct this code based on the Entity Relationship Diagram (ERD) found in the official Synthea documentation. You can also check the example of the tables page in the following link: tables.md.


⚙️ Mkdocs: Dynamic Content with Jinja

To enable dynamic content generation for our documentation site, we’ll use Jinja to process templates and replace placeholders with actual data. Below is a step-by-step breakdown:

  1. Set Up a templates Folder
    Create a folder named templates to store all Markdown files for the site. These files should include placeholders. For instance, in index.md, you might have placeholders like {{database.version_date}} and {{database.version}}.

  2. Utilize Placeholders
    Placeholders are dynamic variables in the Markdown files. These variables will be updated automatically using Python dictionaries to inject relevant data.

  3. Generate Dynamic Content with update.py

    • Prepare your Markdown templates by identifying the sections where dynamic data is required.

    • Use a Python script (update.py), available in my repository, to process the templates. The script performs the following tasks:

      • Database Connection: Connects to a SQLite database to fetch the latest values.
      • Template Rendering: Uses the Jinja library to substitute placeholders with data from the database.
      • File Generation: Outputs updated Markdown files to the docs folder, ready for rendering in MkDocs.

from jinja2 import Template
def render_template(template_str, data):
    """Render the template with the data."""
    template = Template(template_str)
    return template.render(data)
# Data structure
data_dict = {
    'database': {
        'version': 1,
        'version_date': '2024-01-01'
    }
}
# Render the template
rendered_content = render_template("Data updated to {{database.version_date}}", data_dict)
print(rendered_content) 
Data updated to  2024-10-01 10:20:30

By following these steps, you can automate the updating process for your documentation site, ensuring the content remains dynamic and relevant without manual edits.


Dynamic Update of Data Tables

In the next example, we will update the content in the tables.md file to show an example of the persons table from the database. To do this, we will create a placeholder {{table.person}} within the Markdown file. The idea is to dynamically fetch the data from the persons table, and then use the Jinja library along with pandas to convert the query results into a Markdown table format.

Here’s an example of how the tables.md file looks with the placeholder:

#### Example Person Table
## Person Table
{{table.person}}

The process is as follows:

  1. Query the Database: The script will query the persons table in the SQLite database to fetch relevant records.
  2. Convert to Markdown: Using pandas, the results of the query will be converted into a Markdown table format.
  3. Replace the Placeholder: The {{table.person}} placeholder in the tables.md file will be replaced by the generated Markdown table.
import sqlite3
import pandas as pd
import aiosql


def get_queries():
    sql = aiosql.from_path('template/db/queries', 'sqlite3')
    return sql

def get_table_person(db_name):
    """Get a DataFrame from the PATIENTS table."""
    query = get_queries().get_example_patients.sql 
    connection = sqlite3.connect(db_name)

    df = pd.read_sql_query(query, connection)
    connection.close()
    return df

def update_tables_file(template_path, output_dir, db_name):
    """Update the tables.md file"""
    df = get_table_person(db_name)
    # Convert the dataframe in markdown format
    data_dict = {
        'table': {
            'person': df.to_markdown(index=False)  
        }
    }


print(get_table_person("hospital.db"))  

This way, the documentation always reflects up-to-date data, displaying dynamic examples based on the actual content from the database.


⚙️ Mkdocs: Final Workflow

  1. Create Templates: Develop your pages in the docs/template directory.
  2. Run update.py: Populate dynamic content and generate the final files in docs/output.
  3. Preview Locally: Use mkdocs serve to preview the site on localhost.
  4. Build for Deployment: Use mkdocs build to generate a static site in the docs/ folder.
  5. Deploy: Use Terraform to deploy the site to an AWS S3 bucket. Refer to the deployment section of this post for detailed instructions.

I

🦖 Docusaurus Implementation

In the following sections, I will provide detailed steps and insights on how to implement a documentation site using Docusaurus. This includes setup, customization, and deployment options.


🚀 Key Features of Docusaurus


🔧 Docusaurus Setup: From Scratch

To get started with Docusaurus, we follow a quick setup process, which is very similar to the steps we used for MkDocs but with different tools.

  1. Create a New Docusaurus Project: First, install Node.js and run the following command to create a new Docusaurus site:

    npx create-docusaurus@latest my-website classic
  2. Install Mermaid Package: To enable Mermaid diagrams, install the required package:

    npm install @docusaurus/theme-mermaid
  3. Run the Development Server: Once installed, navigate to your project directory and run the development server:

    cd my-website
    npx docusaurus start
  4. Visit the Site: Your site will be live locally at: http://localhost:3000.


🔧 Docusaurus Customization: Configuration

The configuration file docusaurus.config.js is where we customize the title, theme, navigation, and enable features like Mermaid for diagram rendering. Example snippet for enabling Mermaid:

module.exports = {
  title: 'Hospital Documentation',
  tagline: 'Documentation for Hospital Data ML Project',
  favicon: 'img/favicon.ico',
  url: 'https://your-site-url.com',
  markdown: {
    mermaid: true,  // Enable Mermaid diagrams
  },
  themeConfig: {
    navbar: {
      title: 'Hospital Docs',
      items: [
        { to: 'docs/', label: 'Home', position: 'left' },
        { to: 'docs/tables', label: 'Tables', position: 'left' },
        { to: 'docs/architecture', label: 'Architecture', position: 'left' },
        { to: 'docs/glossary', label: 'Glossary', position: 'left' },
      ],
    },
    footer: {
      style: 'dark',
      links: [
        { label: 'GitHub', href: 'https://github.com/your-repo' },
      ],
    },
  },
};

🔧 Docusaurus Customizing the Homepage

To customize the homepage, we modify the src/components/HomepageFeatures/index.js file. Here, you can adjust the FeatureList object to update the features displayed on the homepage.

Image description

📂 Docusaurus Content Organization and Structure

Just like in MkDocs, Docusaurus supports Markdown files for content, and we organize the structure as follows: 1. Template Folder: Store your Markdown files in the docs/template directory, and create a Python script (similar to update.py) to fetch and populate dynamic data into these templates. 2. Category File (__category__.json): To manage the order of documents in the sidebar, create a __category__.json file in each folder. For example:

Architecture
  ├── architecture.md
  ├── img
  └── __category__.json

__category__.json Example: json { "label": "Architecture", "position": 2, "link": { "type": "generated-index", "description": "AWS Data Processing Blueprint" } }

Image description

⚙️ Dynamic Data with Jinja

To incorporate dynamic content, such as database tables, we use a 🐍Python script named update.py, which you can find in the repository.

This script fetches data from a SQLite database and processes the Markdown files stored in the templates folder. It then updates these files with the fetched data and copies them into the docs folder, preparing them for site rendering.

This workflow ensures that the content remains up-to-date and ready for deployment, following a similar approach to what we implemented with MkDocs.


⚙️ Docusaurus: Final Workflow

  1. Create Templates: Develop your Markdown files within the docs/template directory.
  2. Run Python Script: Use the script to dynamically populate data into the templates.
  3. Preview Locally: Run npx docusaurus start to preview the site.
  4. Build for Deployment: Once ready, use npx docusaurus build to generate the static site.
  5. Deploy: Host the static files on your preferred platform, such as AWS S3.
Image description

🚀 Deployment

In this section, we will cover the deployment process for both MkDocs and Docusaurus using AWS S3 for hosting. While the deployment steps are the same for both tools, the installation processes differ, with MkDocs being Python-based and Docusaurus being JavaScript-based.


Infrastructure Setup with Terraform

To deploy a static documentation site to AWS S3, we use Terraform to provision and configure the required resources. The setup defines the S3 bucket, enables static website hosting, and configures public access with a bucket policy to allow read-only access. You can find the main.tf file in the repository.


🚀 Key Components for S3 Deployment

  1. S3 Bucket Creation: The resource to create the S3 bucket where the documentation will be hosted.
  2. Static Website Hosting: Configuration for static web hosting, setting the index.html and error.html as the main and error documents.
  3. Public Access Configuration: Manages public access to the S3 bucket, ensuring it is configured for read-only access.
  4. Bucket Policy: Allows public access to retrieve the documentation content from the S3 bucket.

You can access the complete Terraform file and the corresponding configurations for deploying the site in the repository:

Terraform Configuration File: * mkdocs file * docusaurus file

GitHub Action Workflow for Automatic Deployment: A CI/CD pipeline to automate the deployment process is also included in the repository. * mkdocs file * docusaurus file

GitHub Actions Configuration Make sure to configure your AWS credentials in the GitHub repository secrets under Settings > Secrets > Actions. This will allow GitHub Actions to securely access your AWS account and perform actions like uploading files to S3 when you push changes to the main branch.


Repositories

Below are the links to all the code to deploy your documentation site. If you find it useful, you can leave a star ⭐️ and follow me to receive notifications of new articles. This will help me grow in the tech community and create more content.


🔍 Final Conclusions: MkDocs vs. Docusaurus

Both solutions are easy to implement, but in the following items, we can explore some differences, and what is the best solution depends on the context, knowledge, and complexity you may need to implement.


📚 References

  1. Mkdocs: https://www.mkdocs.org/
  2. Mkdocs-Material: https://squidfunk.github.io/mkdocs-material/
  3. Diagrams: https://diagrams.mingrammer.com/
  4. Docusaurus: https://docusaurus.io/
  5. Jinja: https://jinja.palletsprojects.com/en/stable/
  6. Git Book - What is doc as code: https://www.gitbook.com/blog/what-is-docs-as-code
  7. Write the docs: https://www.writethedocs.org/guide/docs-as-code/

Reuse

Text and figures are licensed under Creative Commons Attribution CC BY 4.0. The figures that have been reused from other sources don't fall under this license and can be recognized by a note in their caption: "Figure from ...".

Citation

For attribution, please cite this work as

Mendez (2025, Jan. 10). Romina Mendez: Deploying Docs-as-Code on AWS: Building Dynamic Documentation Sites in MkDocs and Docusaurus. Retrieved from https://r0mymendez.github.io/posts_en/2025-01-10-deploying-docs-as-code-on-aws-building-dynamic-documentation-sites-in-mkdocs-and-docusaurus/

BibTeX citation

@misc{mendez2025deploying,
  author = {Mendez, Romina},
  title = {Romina Mendez: Deploying Docs-as-Code on AWS: Building Dynamic Documentation Sites in MkDocs and Docusaurus},
  url = {https://r0mymendez.github.io/posts_en/2025-01-10-deploying-docs-as-code-on-aws-building-dynamic-documentation-sites-in-mkdocs-and-docusaurus/},
  year = {2025}
}