Continuous integration#
Doctools package pipeline#
Doctools has a continuous deployment integration pipeline that works as follows:
┌──────────────────┐
┌─►│Build Doc & Deploy├─┐
│ └──────────────────┘ │
│ │
┌─────────────┐ │ ┌────────────────┐ │ ┌──────────────┐
│Build Package├─┼─►│Build Doc on Min├───┼─►│Deploy Package│
└─────────────┘ │ └────────────────┘ │ └──────────────┘
│ │
│ ┌─────┐ │
└─►│Tests├──────────────┘
└─────┘
The Build Package step “compiles” JavaScript and SASS, fetches third-party assets and licenses and generates the Python package.
Then, in the middle-stage, two parallel runs are launched:
Build Doc Latest: uses the latest stable dependencies releases to generate this documentation, and store as an artifact.
Build Doc on Min: uses the minimum requirements dependencies to generate this documentation, but the output is discarded.
Tests: run tests using
pytest
, in special, methods that are not called during the Build Doc * pipelines.
Both of them are set to fail-on-warning during the documentation generation.
Finally, the Deploy Package:
Grabs the version and checks if the tag version already exists:
If so, set to update the symbolic
pre-release
release.If not, set to update the symbolic
latest
andpre-relase
release.
Still if a new version:
Create the git tag and push to origin.
Create the tagged release.
Upload the artifact to the tagged release.
Upload the artifact to the symbolic release (
pre-release
,latest
).Finally, the Build Doc Latest artifact is downloaded and deployed to the branch
gh-pages
By design, the live page on github.io follows the pre-release/latest commit-ish; properly versioned live documentation should be managed by an external system that watches the git tags (e.g. readthedocs).
This approach allows to have a single defined version on adi_doctools/__init__.py
,
and have the tags created and releases created/updated without much fuzz.
The philosophy is to have latest
updated on tag increment and first
successful run, and pre-relese
updated on successful run without tag change.
These releases exist to provide a pointer to the latest/pre-release packages, e.g.
releases/download/latest/adi-doctools.tar.gz.
Non-handled corner-cases mitigations:
Release
pre-release
andlatest
must exist prior the first run.Branch
gh-pages
must exist with at least one commit.
Documentation build and deployment#
Doctools is developed to work offline, in a local server, as a rolling release (e.g. on GitHub Page)s and versioned in a dedicated server (with orchestration).
Local#
For offline and local server, a make html
suffice to generate the
documentation.
To serve the build in a local server, Python built-in server can be used:
~$
python -m http.server -d /path/to/docs/_build/html
Or with hot reload using Author Mode:
~$
adoc author-mode -d /path/to/docs
Rolling release#
For a rolling release, a workflow file is used.
With GitHub Actions, the following workflow file is recommended:
on:
push:
branches:
- main
pull_request:
jobs:
build-doc:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-python@v5
with:
python-version: "3.x"
- name: Install pip packages
working-directory: docs
run: |
pip install pip --upgrade
pip install -r requirements.txt
- name: Build doc
working-directory: docs
run: |
make html SPHINXOPTS='-W --keep-going'
- name: Store the generated doc
uses: actions/upload-artifact@v4
with:
name: html
path: docs/_build/html
deploy-doc:
runs-on: ubuntu-latest
needs: build-doc
if: github.ref == 'refs/heads/main'
steps:
- run: |
git config --global user.name "${{ github.event.head_commit.committer.name }}"
git config --global user.email "${{ github.event.head_commit.committer.email }}"
- uses: actions/checkout@v4
- name: Create gh-pages branch
run: >
git ls-remote --exit-code --heads origin refs/heads/gh-pages ||
(
git reset --hard ;
git clean -fdx ;
git checkout --orphan gh-pages ;
git reset --hard;
git commit -m "empty" --allow-empty ;
git push origin gh-pages:gh-pages
)
- uses: actions/checkout@v4
with:
ref: 'gh-pages'
- name: Empty gh-pages
run: |
git rm -r . --quiet || true
- uses: actions/download-artifact@v4
with:
name: html
- name: Patch doc build
run: |
rm -r _sources
touch .nojekyll
- name: Commit gh-pages
run: |
git add . >> /dev/null
git commit -m "deploy: ${GITHUB_SHA}" --allow-empty
- name: Push to gh-pages
run: |
git push origin gh-pages:gh-pages
With the Sphinx -W
flag, Sphinx exits with an error if any warning is logged,
and --keep-going
continues the build even if a warning is logged, to provide
a complete log for analysis.
The deploy-doc
job only runs when push/merged to main.
Attention
GITHUB_SHA
on pull_request
is the pre-commit and not the head commit,
please be aware of GitHub events values if implementing something else.
The requirements.txt file should contain:
sphinx
https://github.com/analogdevicesinc/doctools/releases/download/latest/adi-doctools.tar.gz
Versioned#
The live versioned version requires additional orchestration than the
Rolling release.
Store each version in separated folders in the root path, e.g.
v1.1
, v2.2
, main
, dev
and generate a tags.json
containing a plain array with each version on the root path,
e.g. ["v1.1", "v2.2", "main", "dev"]
(a empty string means there is a built doc on the root).
This tags.json
is used solely by the opened live page to show a dropdown to
switch between versions.
Tip
See this repo’s .github/workflows/deploy.yml for a suggestion on how to implement it.
For In organization reference, the doc shall target a specific version by suffixing
the target the version on the interref_repos
variable, e.g.
interref_repos = ['pyadi-iio/dev', 'other-repo/v1.1']
.
A basic tags.json
can be obtained with:
~$
ls -d */ | cut -f1 -d'/' | jq --raw-input . | jq --slurp . > tags.json
That means, take all directories in the root path and store as JSON.
If the root does not contain any built doc, add a redirect HTML file pointing to the main/stable version:
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="refresh" content="0; url=main/index.html" />
</head>
</html>
If the root does contain a built doc, add an empty string ""
to
tags.json
.
This entry will be named latest
in the version selector dropdown.
Finally, set the depth of the destination path during doc generation
with ADOC_TARGET_DEPTH
, for example, if the target directory is:
./:
0
or unset./v2.2:
1
./prs/1234:
2
e.g. ADOC_TARGET_DEPTH=2 make html