Jacob Tomlinson's profile picture Jacob Tomlinson
Home Blog Talks Newsletter About

Documenting Python projects with Sphinx and Read the Docs

7 minute read #python, #github, #tutorial

This is part 9 in the Creating an open source Python project from scratch series. If you haven’t read the previous parts you may want to go back and check those out.

In part four of this series we discussed documenting our code as we went along by adding docstrings throughout out project. In this post we will see that effort pay off by building a documentation site using Sphinx which will leverage all of our existing docstrings.

Sphinx

Sphinx is a documentation generation tool which takes in reStructuredText files along with docstrings from within your code and builds a static HTML/CSS documentation website for you.

Let’s get started by installing Sphinx.

$ pip install sphinx

Now we can use the sphinx-quickstart command to create some scaffolding for us.

$ mkdir docs
$ cd docs
$ sphinx-quickstart
$ cd ../

We will be prompted to answer a few questions about the name of the project, authors names, etc. Fill in the required fields and leave the rest as defaults. You should now see a few files have been created for you in the docs directory, let’s talk through each of these:

One thing I also like to do at this point is to create a requirements_docs.txt file in our docs directory to capture all of our documentation dependencies. We’ve already installed sphinx so we should include that, but lets also install sphinx-autobuild to simplify building and testing out our documentation instead of having to use the make files.

# docs/requirements_docs.txt
sphinx
sphinx-autobuild
$ pip install -r docs/requirements_docs.txt

Now we can build and serve our documentation.

$ sphinx-autobuild docs docs/_build/html

+--------- manually triggered build ---------------------------------------------
| Running Sphinx v3.5.3
| building [mo]: targets for 0 po files that are out of date
| building [html]: targets for 1 source files that are out of date
| updating environment: [new config] 1 added, 0 changed, 0 removed
| reading sources... [100%] index
|
| looking for now-outdated files... none found
| pickling environment... done
| checking consistency... done
| preparing documents... done
| writing output... [100%] index
|
| generating indices... genindex done
| writing additional pages... search done
| copying static files... done
| copying extra files... done
| dumping search index in English (code: en)... done
| dumping object inventory... done
| build succeeded.
|
| The HTML pages are in docs/_build/html.
+--------------------------------------------------------------------------------

[I 210331 16:06:29 server:296] Serving on http://127.0.0.1:8000
[I 210331 16:06:29 handlers:62] Start watching changes
[I 210331 16:06:29 handlers:64] Start detecting changes

Now if we navigate to http://127.0.0.1:8000 in our browser we should be able to see our documentation.

Hello world sphinx documentation site

Content

With sphinx-autobuild running it will be watching our docs directory for changes and will rebuild out site automatically. It will also refresh the page for us in our browser. So let’s start adding some content.

We are going to look at two types of documentation here; narrative and autodoc.

Narrative

Narrative documentation is prose written out to communicate high level concepts about your project. What is it? What does it do? Who made it? When should you use it?

To get started I’m going to open up docs/index.rst and copy in the badges, description and installation instructions from our README.rst file and throw in a single usage example. I’m also going to tidy up some of the boilerplate that was generated for me, just leaving the toctree section which lays our our table of contents in the sidebar.

is-number
=========

.. image:: https://img.shields.io/pypi/v/is-number
   :target: https://pypi.org/project/is-number/
   :alt: PyPI
.. image:: https://github.com/jacobtomlinson/is-number/workflows/CI/badge.svg
   :target: https://github.com/jacobtomlinson/is-number/actions?query=workflow%3ACI
   :alt: GitHub Actions - CI
.. image:: https://github.com/jacobtomlinson/is-number/workflows/pre-commit/badge.svg
   :target: https://github.com/jacobtomlinson/is-number/actions?query=workflow%3Apre-commit
   :alt: GitHub Actions - pre-commit
.. image:: https://img.shields.io/codecov/c/gh/jacobtomlinson/is-number
   :target: https://app.codecov.io/gh/jacobtomlinson/is-number
   :alt: Codecov

A Python library to determine if something is a number.

Installation
------------

.. code-block:: bash

   pip install is-number


Usage
-----

.. code-block:: python

   >>> from is_number import is_number
   >>> is_number(10)
   True
   >>> is_number("hello")
   False


.. toctree::
   :maxdepth: 2
   :caption: Contents:

What I’m trying to do here is strike a balance between giving the user enough information to decide they want to use my package, but balance that with overwhelming them with too much text.

Index page copied from README

One small thing to note here is that I’ve included two code examples here in my narrative documentation, so these will not be covered by my doctests. If I change the API then it may invalidate the documentation page by making the information incorrect.

I personally like to keep code examples short on the landing page to hopefully minimise this, but I think it is important to show some code to users when they first arrive.

At the end of the day your narrative documentation is a large part of your project’s marketing.

Autodoc documentation

Now that we have a basic landing page let’s make use of those docstrings we wrote earlier in the series. Using doctrings with examples that are tested is great because it ensures your documentation pages will not get out of date with out code as you make changes to the project.

In order for sphinx to include our docstrings will be making use of the sphinx.ext.autodoc plugin.

To enable autodoc we need to add it to our list of loaded extensions in our docs/conf.py. Find the empty extensions list and add it.

extensions = [
    "sphinx.ext.autodoc",
]

Now let’s create a new file called docs/api.rst to store our autogenerated documentation on the is_number API.

API
===

``is-number``
-------------

.. autofunction:: is_number.is_number

``is-float``
-------------

.. autofunction:: is_number.is_float

Here we have created titles to set the layout for the page but for the actual content we’ve used the autofunction directive which will include the doctring from that function.

Lastly we need to update the toctree in our index.rst file to include this new page.

.. toctree::
   :maxdepth: 2
   :caption: Contents:

   api

Now if we head back to our browser we should be able to click the new API link on the sidebar and see our auto generated page.

New API documentation page

Given our is-number project is pretty simple that’s probably enough here. For your project you may want to add more pages and sections, perhaps your install is more complex and deserves it’s own page.

Read the Docs

Now that we have successfully created our documentation site we will want to host it somewhere on the internet for our users to view.

A great service for this is Read the Docs.

Read the Docs

Head to that website and sign up for an account (I just log in with my GitHub account).

Then head to the import a project page so we can import our GitHub repo.

Add a project in RtD

The project information will be populated for us so we can just go ahead and continue.

is-number on RtD

Now that we have our project set up let’s also add a .readthedocs.yml config file to our project.

version: 2

sphinx:
  configuration: docs/conf.py

formats: all

python:
  version: 3.7
  install:
    - method: pip
      path: .
      extra_requirements:
        - all
    - requirements: docs/requirements_docs.txt

submodules:
  include: all

Now that we have everything configured let’s commit everything to a new branch and make a pull request.

$ git checkout -b add-docs
$ git add -A
$ git commit -m "Add Sphinx documentation"  # Our pre-commit checks may not be happy with the sphinx conf.py and format it and you'll need to add and commit again
$ git push --set-upstream origin add-docs

Add docs pull request

Now if we merge our PR Read the Docs will build our documentation for us and make it available at https://is-number.readthedocs.io/.

Our docs live on Read the Docs

In exchange for a small advert displayed on our documentation we now have free hosting for our documentation website.

Summary

In this post we have covered:

In future posts we will cover:

More in this series

This was part 9 in the Creating an open source Python project from scratch series.

Next up, Building a user community for your open source project

Be sure to subscribe to the RSS feed to hear about future parts in this series.


Have thoughts?

I love hearing feedback on my posts. You should head over to Twitter and let me know what you think!

Spotted a mistake? Why not suggest an edit!