This guide is intended for developers of the AiiDAlab platform itself and covers topics such as the overall architecture, release workflow, etc.
Key AiiDAlab components¶
The key components of AiiDAlab are:
AiiDAlab package provides the core functionality of the service.
AiiDAlab home app provides the web-interface to the service
AiiDAlab app registry registry for apps shared by the AiiDAlab community.
The aiidalab package is used to implement the back-end application logic for the AiiDAlab environment. This includes the installation and removal of apps, their discovery, and the app store.
An app is a collection of Jupyter notebooks (with optional dependencies which are either directly vendored or declared as dependencies in a
In the current implementation status, those dependencies must be pre-installed and are not installed with the app!
The aiidalab-home package implements the front-end for the AiiDAlab environment in the form of the so called home app. That includes for example the start page, the terminal, the app store, and the app management interfaces. The aiidalab-home package depends on the aiidalab package. Unlike other apps, it is not intended for users to uninstall the home app, or change the installed version. That is to ensure that the AiiDAlab environment remains functional for a specific docker image and to make it possible to release backwards incompatible changes to the environment (e.g. a new version of the aiidalab package or other dependencies), without breaking existing user environments.
The aiidalab-home app is installed into the system environment and then linked into the user’s user space via a symbolic link. This ensures that the system package is largely immutable but is still available as an app in user space. For development purposes, developers can remove the symbolic link and replaces it with a different version of aiidalab-home; an eventual user installation of aiidalab-home will be automatically moved to backup location and replaced with the symbolic link upon updating the environment’s docker image.
The AiiDAlab environment is released and deployed through the aiidalab-docker-stack Docker images. Those images install specific versions of aiidalab, aiidalab-home, and other system-wide dependencies, that are expected to be compatible. Continuous integration tests for compatibility are executed upon each commit of the aiidalab-docker-stack image.
Please see the documentation of each individual component for instructions on how exactly to implement and execute tests.
The aiidalab package is tested via unit and integration tests for function and compatibility. All tests are targeted at ensuring that new development does not inadvertently break existing behavior and that new features behave as expected.
The aiidalab-home package is tested via front-end tests using the aiidalab-test-app-action. Those tests are targeted at ensuring that the home app is functional on the latest development version of the aiidalab-docker-stack. This means that the introduction of incompatibilities must be addressed after updating the aiidalab-docker-stack development version.
The aiidalab-docker-stack is tested via front-end tests that are implemented as part of a select set of tested applications, including the aiidalab-home app. This is to ensure that these apps work as intended when installed.
AiiDAlab and AiiDAlab Home app development¶
For the following instructions, it is assumed that any development is directly performed within the AiiDAlab Terminal environment.
The steps to setup (and teardown) an AiiDAlab development workflow are automated within the aiidalab-dev package.
Setup for front-end development (home app)¶
In your development environment (local or in the cloud), remove the symbolic link of aiidalab-home (
/home/aiida/apps/home) and replace it with a link to a clone of your fork of the aiidalab-home app.
Then commence development on a new branch:
~$ git clone firstname.lastname@example.org:<USERNAME>/aiidalab-home.git ~$ cd apps/ ~/apps$ rm home # DO NOT USE `rm -r`! ~/apps$ ln -s ~/aiidalab-home/ home
Once satisfied with the change set, push it to your fork and create a (draft) pull request. This will ensure that the front-end tests are executed against the latest development version of aiidalab-docker-stack. Address all issues and then release a new version.
Setup for back-end development (aiidalab package)¶
First, replace the system home app with a local copy based on your fork as described in the instructions for development of the home app.
The clone your fork of the aiidalab package into a location within user space of your choice (e.g.
$HOME/aiidalab) and create a symbolic link within the home app to that location:
~$ git clone email@example.com:<USERNAME>/aiidalab.git ~$ cd apps/home/ ~/apps/home$ ln -s $HOME/aiidalab/aiidalab
In this way, you can test the function of the home app directly against your development version of aiidalab.
Once satisfied with the change set, push it to your fork and create a (draft) pull request. This will ensure that the unit and integration tests are executed.
How to make a new release (aiidalab and aiidalab-home)¶
Make a new release branch e.g.
$ git checkout -b release/1.0.0a2
Commit the changes with the commit message: “Prepare release 1.0.0a2”.
Push the changes to GitHub.
Create a PR from
release/1.0.0a2branch to the
Once the tests are passed, merge the PR via merge commit (do not use rebase, it will recreate all the commits’ sha!).
Go to the command line and pull the
$ git branch -d release/1.0.0a2
Make a new annotated tag. Commit message: “Release 1.0.0a2”
$ git tag -a 1.0.0a2
Push the new tag to GitHub:
$ git push --tags
List all new changes (with links to PRs) while making a release on GitHub.
Remove the old dist folder.
$ rm -r dist
Create a new
$ python setup.py sdist
Upload things on test.pypi.org
$ twine upload --repository-url https://test.pypi.org/legacy/ dist/*
Check on test.pypi.org that everything is alright.
Upload things on pypi.org:
$ twine upload dist/*
developand push the
developbranch to GitHub.
$ git checkout develop && git merge master && git push