Documentation
In today's rapidly evolving software development landscape, effective documentation is a crucial component of any project. The ability to create clear, concise, and user-friendly technical documentation can make a significant difference in the success of your codebase. We all probably encountered code that we wanted to use, only for us to abandon using it because it was missing documentation such that we could get started with it.
Technical documentation or code documentation can be many things:
- Plain text, images and videos explaining core concepts for your software
- Documentation of API on how to call a function or class, what the different parameters are etc.
- Code examples of how to use certain functionality
and many more. We are in this module going to focus on setting up a very basic documentation system that will automatically help you document the API of your code. For this reason we recommend that before continuing with this module that you have completed module M7 on good coding practices or have similar experience with writing docstrings for Python functions and classes.
There are different systems for writing documentation. In fact there is a lot to choose from:
Important to note that all these are static site generators. The word static here refers to that when the content is generated and served on a webside, the underlying HTML code will not change. It may contain HTML elements that dynamic (like video), but the site does not change (1).
- Good examples of dynamic sites are any social media or news media where new posts, pages etc. are constantly added over time. Good examples of static sites are documentation, blogposts etc.
We are in this module going to look at Mkdocs, which (in my opinion) is one of the easiest systems to get started with because all documentation is written in markdown and the build system is written in Python. As an alternativ, you can consider doing the exercises in Sphinx which is probably the most used documentation system for Python code. Sphinx offer more customization than Mkdocs, so is generally preferred for larger projects with complex documentation, but for smaller projects Mkdocs should be easier to get started with and is sufficient.
Mkdocs by default does not include many features and for that reason we are directly going to dive into using the material for mkdocs theme that provides a lot of nice customization to create professional static sites. In fact, this whole course is written in mkdocs using the material theme.
Mkdocs
The core file when using mkdocs is the mkdocs.yaml
file, which is the configuration file for the project:
site_name: Documentation of my project
site_author: Jane Doe
docs_dir: source # (1)!
theme:
language: en
name: material # (2)!
features: # (3)!
- content.code.copy
- content.code.annotate
plugins: # (4)!
- search
- mkdocstrings
nav: # (5)!
- Home: index.md
-
This indicates the source directory of our documentation. If the layout of your documentation is a bit different than what described above, you may need to change this.
-
The overall theme of your documentation. We recommend the
material
theme but there are many more to choose from and you can also create your own. -
The
featuers
section is where features that are supported by your given theme can be enabled. In this example we have enabledcontent.code.copy
feature which adds a small copy button to all code block and thecontent.code.annotate
feature which allows you to add annotations like this box to code blocks. -
Plugins add new functionality to your documentation. In this case we have added two plugins that add functionality for searching through our documentation and automatically adding documentation from docstrings. Remember that some plugins requires you to install additional Python packages with those plugins, so remember to add them to your
requirements.txt
file. -
The
nav
section is where you define the navigation structure of your documentation. When you add new.md
files to thesource
folder you then need to add them to thenav
section.
And that is more or less what you need to get started. In general, if you need help with configuration of your documentation in mkdocs I recommend looking at this page and this page.
Exercises
In this set of exercises we assume that you have completed module M6 on code structure and therefore have a repository that at least contains the following:
├── pyproject.toml <- Project configuration file with package metadata
│
├── docs <- Documentation folder
│ │
│ ├── index.md <- Homepage for your documentation
│ │
│ ├── mkdocs.yaml <- Configuration file for mkdocs
│ │
│ └── source/ <- Source directory for documentation files
│
└── src <- Source code for use in this project.
│ │
│ ├── __init__.py <- Makes src a Python module
│ │
│ ├── models <- model implementations, training script
│ │ ├── __init__.py
│ │ ├── model.py
│ │ ├── train_model.py
...
It is not important exactly what is in the src
folder for the exercises, but we are going to refer to the above
structure in the exercises, so adjust accordingly if you diviate from this. Additionally, we are going to assume that
your project code is installed in your environment such that it can be imported as normal Python code.
-
We are going to need two Python packages to get started: mkdocs and material for mkdocs. Install with
- Since
mkdocs
is a dependency ofmkdocs-material
we only need to install the latter.
- Since
-
Run in your terminal (from the
docs
folder):-
mkdocs serve
will automatically rebuild the whole site whenever you save a file inside thedocs
folder. This is not a problem if you have a fairly small site with not that many pages (or elements), but can take a long time for large sites. Consider running with the--dirty
option for only re-building the site for files that have been changed.
which should render the
index.md
file as the homepage. You can leave the documentation server running during the remaining exercises. -
-
We are no ready to document the API of our code:
-
Make sure you at least have one function and class inside your
src
module. If you do not have you can for simplicity copy the following module to thesrc/models/model.py
fileimport torch class MyNeuralNet(torch.nn.Module): """Basic neural network class. Args: in_features: number of input features out_features: number of output features """ def __init__(self, in_features: int, out_features: int) -> None: self.l1 = torch.nn.Linear(in_features, 500) self.l2 = torch.nn.Linear(500, out_features) self.r = torch.nn.ReLU() def forward(self, x: torch.Tensor) -> torch.Tensor: """Forward pass of the model. Args: x: input tensor expected to be of shape [N,in_features] Returns: Output tensor with shape [N,out_features] """ return self.l2(self.r(self.l1(x)))
and the following function to add
src/predict_model.py
file:def predict( model: torch.nn.Module, dataloader: torch.utils.data.DataLoader ) -> None: """Run prediction for a given model and dataloader. Args: model: model to use for prediction dataloader: dataloader with batches Returns Tensor of shape [N, d] where N is the number of samples and d is the output dimension of the model """ return [model(batch) for batch in dataloader]
-
Add a markdown file to the
docs/source
folder calledmy_api.md
and add that file to thenav:
section in themkdocs.yaml
file. -
To that file add the following code:
The
:::
indicator tells mkdocs that it should look for the corresponding function/module and then render it on the given page. Thus, if you have a function/module located in another location change the paths accordingly. -
Make sure that the documentation correctly includes your function and module on the given page.
-
(Optional) Include more functions/modules in your documentation.
-
-
(Optional) Look through the documentation for mkdocstrings and try to improve the layout a bit. Especially, the headings, docstrings and signatures could be of interest to adjust.
-
Finally, try to build a final version of your documentation
this should result in a
site
folder that contains the actual HTML code for documentation.
Publish your documentation
To publish your documentation you need a place to host your build documentation e.g. the content of the site
folder
you build in the last exercise. There are many places to host your documentation, but if you only need a static site
and are already hosting your code through Github, then a good option is Github Pages.
Github pages is free to use for your public projects.
Before getting started with this set of exercises you should have completed module M16 on GitHub actions so you already know about workflow files.
Exercises
-
Start by adding a new file called
deploy_docs.yaml
to the.github/workflows
folder. Add the following cod to that file and save it.name: Deploy docs on: push: branches: - main permissions: contents: write # (1) jobs: deploy: name: Deploy docs runs-on: ubuntu-latest steps: - name: Checkout code uses: actions/checkout@v4 with: fetch-depth: 0 - name: Set up Python uses: actions/setup-python@v5 with: python-version: 3.11 cache: 'pip' cache-dependency-path: setup.py - name: Install dependencies run: pip install -r requirements.txt - name: Deploy docs run: mkdocs gh-deploy --force
- It is important to give
write
permissions to this actions because it is not only reading your code but it will also push code.
Before continuing, make sure you understand what the different steps of the workflow does and especially we recommend looking at the documentation of the
mkdocs gh-deploy
command. - It is important to give
-
Commit and push the file. Check that the action is executed and if it succeeds, that your build project is pushed to a branch called
gh-pages
. If the action does not succeeds, then figure out what is wrong and fix it! -
After confirming that our action is working, you need to configure Github to publish the content being build by Github Actions. Do the following:
- Go to the Settings tab and then the Pages subsection
- In the
Source
setting choose theDeploy from a branch
- In the
Branch
setting choose thegh-pages
branch and/(root)
folder and save
This should then start deploying your site to
https://<your-username>.github.io/<your-reponame>/
. If it does not do this you may need to recommit and trigger the GitHub actions build again. -
Make sure your documentation is published and looks as it should.
This ends the module on technical documentation. We cannot stress enough how important it is to write proper documentation for larger projects that need to be maintained over a longer time. It is often a iterative process, but it is often best to do it while writing the code.