To work in the framework itself you will need Python >= 3.8. Linting, testing,
and docs automation is performed using
tox
, which you should install.
For improved performance on the tests, ensure that you have PyYAML
installed with the correct extensions:
apt-get install libyaml-dev
pip install --force-reinstall --no-cache-dir pyyaml
The following are likely to be useful during development:
# Run linting and unit tests
tox
# Run tests, specifying whole suite or specific files
tox -e unit
tox -e unit test/test_charm.py
# Format the code using Ruff
tox -e fmt
# Compile the requirements.txt file for docs
tox -e docs-deps
# Generate a local copy of the Sphinx docs in docs/_build
tox -e docs
# run only tests matching a certain pattern
tox -e unit -- -k <pattern>
For more in depth debugging, you can enter any of tox
's created virtualenvs
provided they have been run at least once and do fun things - e.g. run
pytest
directly:
# Enter the linting virtualenv
source .tox/lint/bin/activate
...
# Enter the unit testing virtualenv and run tests
source .tox/unit/bin/activate
pytest
...
The framework has some tests that interact with a real/live Pebble server. To
run these tests, you must have pebble
installed and available in your path. If you have the Go toolchain installed,
you can run go install github.com/canonical/pebble/cmd/pebble@master
. This will
install pebble to $GOBIN
if it is set or $HOME/go/bin
otherwise. Add
$GOBIN
to your path (e.g. export PATH=$PATH:$GOBIN
or export PATH=$PATH:$HOME/go/bin
in your .bashrc
) and you are ready to run the real
Pebble tests:
tox -e pebble
To do this even more manually, you could start the Pebble server yourself:
export PEBBLE=$HOME/pebble
export RUN_REAL_PEBBLE_TESTS=1
pebble run --create-dirs --http=:4000 &>pebble.log &
# Then
tox -e unit -- test/test_real_pebble.py
# or
source .tox/unit/bin/activate
pytest -v test/test_real_pebble.py
When making changes to ops
, you'll commonly want to try those changes out in
a charm.
If your changes are in a Git branch, you can simply replace your ops
version
in requirements.txt
(or pyproject.toml
) with a reference to the branch, like:
#ops ~= 2.9
git+https://github.com/{your-username}/operator@{your-branch-name}
git
is not normally available when charmcraft
is packing the charm, so you'll
need to also tell charmcraft
that it's required for the build, by adding
something like this to your charmcraft.yaml
:
parts:
charm:
build-packages:
- git
If your changes are only on your local device, you can inject your local ops
into the charm after it has packed, and before you deploy it, by unzipping the
.charm
file and replacing the ops
folder in the virtualenv. This small
script will handle that for you:
#!/usr/bin/env bash
if [ "$#" -lt 2 ]
then
echo "Inject local copy of Python Operator Framework source into charm"
echo
echo "usage: inject-ops.sh file.charm /path/to/ops/dir" >&2
exit 1
fi
if [ ! -f "$2/framework.py" ]; then
echo "$2/framework.py not found; arg 2 should be path to 'ops' directory"
exit 1
fi
set -ex
mkdir inject-ops-tmp
unzip -q $1 -d inject-ops-tmp
rm -rf inject-ops-tmp/venv/ops
cp -r $2 inject-ops-tmp/venv/ops
cd inject-ops-tmp
zip -q -r ../inject-ops-new.charm .
cd ..
rm -rf inject-ops-tmp
rm $1
mv inject-ops-new.charm $1
If your ops
change relies on a change in a Juju branch, you'll need to deploy
your charm to a controller using that version of Juju. For example, with microk8s:
- Build Juju and its dependencies
- Run
make microk8s-operator-update
- Run
GOBIN=/path/to/your/juju/_build/linux_amd64/bin:$GOBIN /path/to/your/juju bootstrap
- Add a model and deploy your charm as normal
We rely on automation to update charm pins of a bunch of charms that use the operator framework. The script can be run locally too.
Changes are proposed as pull requests on GitHub.
For coding style, we follow PEP 8 as well as a team Python style guide. Please be complete with docstrings and keep them informative for users, as the ops library reference is automatically generated from Python docstrings.
For more advice about contributing documentation, see Contributing documentation.
Pull requests should have a short title that follows the conventional commit style using one of these types:
- chore
- ci
- docs
- feat
- fix
- perf
- refactor
- revert
- test
At present, we only add a scope in these cases:
- If the PR is limited to changes in ops/_private/harness.py, also include the scope
(harness)
- If the PR is limited to changes in testing/, also include the scope
(testing)
For example:
- feat: add the ability to observe change-updated events
- fix!: correct the type hinting for config data
- docs(harness): clarify the types of exceptions that Harness.add_user_secret may raise
- ci(testing): adjust the workflow that publishes ops-scenario
Note that the commit messages to the PR's branch do not need to follow the
conventional commit format, as these will be squashed into a single commit to main
using the PR title as the commit message.
The format for copyright notices is documented in the LICENSE.txt. New files should begin with a copyright line with the current year (e.g. Copyright 2024 Canonical Ltd.) and include the full boilerplate (see APPENDIX of LICENSE.txt). The copyright information in existing files does not need to be updated when those files are modified -- only the initial creation year is required.
The published docs at ops.readthedocs.io
are built automatically from the top-level docs
directory. We use MyST Markdown
for most pages and arrange the pages according to Diátaxis.
To contribute docs:
- Fork this repo and edit the relevant source files:
- Tutorials -
/docs/tutorial
- How-to guides -
/docs/howto
- Reference - Automatically generated from Python docstrings
- Explanation -
/docs/explanation
- Tutorials -
- Build the documentation locally, to check that everything looks right
- Propose your changes using a pull request
When you create the pull request, GitHub automatically builds a preview of the docs. To find the preview, look for the "docs/readthedocs.org:ops" check near the bottom of the pull request page, then click Details. You can use the preview to double check that everything looks right.
- Use short sentences, ideally with one or two clauses.
- Use headings to split the doc into sections. Make sure that the purpose of each section is clear from its heading.
- Avoid a long introduction. Assume that the reader is only going to scan the first paragraph and the headings.
- Avoid background context unless it's essential for the reader to understand.
Recommended tone:
- Use a casual tone, but avoid idioms. Common contractions such as "it's" and "doesn't" are great.
- Use "we" to include the reader in what you're explaining.
- Avoid passive descriptions. If you expect the reader to do something, give a direct instruction.
To build the docs and open them in your browser:
tox -e docs
open docs/_build/html/index.html
Alternatively, to serve the docs locally and automatically refresh them whenever you edit a file:
tox -e docs-live
We don't publish separate documentation for separate versions of ops. The published docs at ops.readthedocs.io are always for the in-development (main branch) of ops, and do not include any notes indicating changes or additions across ops versions. We encourage all charmers to promptly upgrade to the latest version of ops, and to refer to the release notes and changelog for learning about changes.
We do note when features behave differently when using different versions of Juju.
In docstrings:
- Use
.. jujuadded:: x.y
to indicate that the feature is only available when using version x.y (or higher) of Juju. - Use
..jujuchanged:: x.y
when the feature's behaviour in ops changes. - Use
..jujuremoved:: x.y
when the feature will be available in ops but not in that version (or later) of Juju.
Similar directives also work in MyST Markdown. For example:
```{jujuadded} x.y
Summary
```
Unmarked features are assumed to work and be available in the current LTS version of Juju.
The documentation uses Canonical styling which is customised on top of the Furo Sphinx theme. The easiest way to pull in Canonical style changes is by using the Canonical documentation starter pack, see docs and repository.
TL;DR:
- Clone the starter pack repository to a local directory:
git clone git@github.com:canonical/sphinx-docs-starter-pack
. - Copy the folder
.sphinx
under the starter pack repo to the operator repodocs/.sphinx
.
There are two configuration files: docs/conf.py
and docs/custom_conf.py
, copied and customised from the starter pack repo.
To customise, change the file docs/custom_conf.py
only, and theoretically, we should not change docs/conf.py
(however, some changes are made to docs/conf.py
, such as adding autodoc, PATH, fixing issues, etc.)
The Canonical documentation starter pack uses Make to build the documentation, which will run the script docs/.sphinx/build_requirements.py
and generate a requirement file requirements.txt
under docs/.sphinx/
.
To pull in new dependency changes from the starter pack, change to the starter pack repository directory, and build with the following command. This will create a virtual environment, generate a dependency file, install the software dependencies, and build the documentation:
make html
Then, compare the generated file .sphinx/requirements.txt
and the project.optional-dependencies.docs
section of pyproject.toml
and adjust the pyproject.toml
file accordingly.
The Python dependencies of ops
are kept as minimal as possible, to avoid
bloat and to minimise conflict with the charm's dependencies. The dependencies
are listed in pyproject.toml in the project.dependencies
section.
Test environments are managed with tox and executed with pytest, with coverage measured by coverage. Static type checking is done using pyright, and extends the Python 3.8 type hinting support through the typing_extensions package.
Formatting uses Ruff.
All tool configuration is kept in project.toml. The list of
dependencies can be found in the relevant tox.ini
environment deps
field.
The build backend is setuptools, and the build frontend is build.
To make a release of the ops
and/or ops-scenario
packages, do the following:
- Check if there's a
chore: update charm pins
auto-generated PR in the queue. If it looks good, merge it and check that tests still pass. If needed, you can re-trigger theUpdate Charm Pins
workflow manually to ensure latest charms and ops get tested. - Visit the releases page on GitHub.
- Click "Draft a new release"
- The "Release Title" is the full version numbers of ops and/or ops-scenario,
in the form
ops <major>.<minor>.<patch> and ops-scenario <major>.<minor>.<patch>
and a brief summary of the main changes in the release. For example:2.3.12 Bug fixes for the Juju foobar feature when using Python 3.12
- Have the release create a new tag, in the form
<major>.<minor>.<patch>
forops
andscenario-<major>.<minor>.<patch>
forops-scenario
. If releasing both packages, use the ops tag. - If the last release was for both
ops
andops-scenario
, leave the previous tag choice onauto
. If the last release was for only one package, change the previous tag to be the last time the same package(s) were being released. - Use the "Generate Release Notes" button to get a copy of the changes into the notes field. The 'Release Documentation' section below details the form that the release notes and changelog should take.
- For
ops
, change version.py'sversion
to the appropriate string. Forops-scenario
, change the version in testing/pyproject.toml. Both packages use semantic versioning, and adjust independently (that is: ops 2.18 doesn't imply ops-scenario 2.18, or any other number). - Run
tox -e docs-deps
to recompile therequirements.txt
file used for docs (in case dependencies have been updated inpyproject.toml
). - Add, commit, and push, and open a PR to get the changelogs, version bumps, and doc requirement bumps into main (and get it merged).
- Save the release notes as a draft, and have someone else in the Charm-Tech team proofread the release notes.
- If the release includes both
ops
andops-scenario
packages, then push a new tag in the formscenario-<major>.<minor>.<patch>
. This is done by executinggit tag scenario-x.y.z
, thengit push upstream tag scenario-x.y.z
locally (assuming you have configuredcanonical/operator
as a remote namedupstream
). - When you are ready, click "Publish". GitHub will create the additional tag.
Pushing the tags will trigger automatic builds for the Python packages and publish them to PyPI (ops and ops-scenario) (authorisation is handled via a Trusted Publisher relationship). Note that it sometimes take a bit of time for the new releases to show up.
See .github/workflows/publish-ops.yaml and .github/workflows/publish-ops-scenario.yaml for details. (Note that the versions in the YAML refer to versions of the GitHub actions, not the versions of the ops library.)
You can troubleshoot errors on the Actions Tab.
-
Open a PR to change the version strings to the expected next version, with ".dev0" appended (for example, if 3.14.1 is the next expected version, use
'3.14.1.dev0'
).
We produce several pieces of documentation for ops
and ops-scenario
releases, each serving a separate purpose and covering a different level.
Avoid using the word "Scenario", preferring "unit testing API" or "state
transition testing". Users should install ops-scenario
with
pip install ops[testing]
rather than using the ops-scenario
package name
directly.
git log
is used to see every change since a previous release. Obviously, no
special work needs to be done so that this is available. A link to the GitHub
view of the log will be included at the end of the GitHub release notes when
the "Generate Release Notes" button is used, in the form:
**Full Changelog**: https://github.com/canonical/operator/compare/2.17.0...2.18.0
These changes include both ops
and ops-scenario
. If someone needs to see
changes only for one of the packages, then the /testing/
folder can be
filtered in/out.
A changelog is kept in version control that simply lists the changes in each
release, other than chores. The changelog for ops
is at the top level, in CHANGES.md, and the changelog for
ops-scenario
is in the /testing
folder, CHANGES.md.
There will be overlap between the two files, as many PRs will include changes to
common infrastructure, or will adjust both ops
and also the testing API in
ops-scenario
.
Adding the changes is done in preparation for a release. Use the "Generate Release Notes" button in the GitHub releases page, and copy the text to the CHANGES.md files.
- Group the changes by the commit type (feat, fix, and so on) and use full names ("Features", not "feat", "Fixes", not "fix") for group headings.
- Remove any chores.
- Remove any bullets that do not apply to the package. For instance, if a bullet
only affects
ops[testing]
, don't include it in CHANGES.md when doing anops
release. The bullet should go in testing/CHANGES.md instead. Ifops[testing]
is not being released yet, put the bullet in a placeholder section at top of testing/CHANGES.md. - Strip the commit type prefix from the bullet point, and capitalise the first word.
- Strip the username (who did each commit) if the author is a member of the Charm Tech team.
- Replace the link to the pull request with the PR number in parentheses.
- Where appropriate, collapse multiple tightly related bullet points into a single point that refers to multiple commits.
- Where appropriate, add backticks for code formatting.
For example: the PR
* docs: clarify where StoredState is stored by @benhoyt in https://github.com/canonical/operator/pull/2006
is added to the "Documentation" section as:
* Clarify where StoredState is stored (#2006)
The GitHub release notes include the list of changes found in the changelogs, but:
- If both
ops
andops-scenario
packages are being released, include all the changes in the same set of release notes. If only one package is being released, remove any bullets that apply only to the other package. - The links to the PRs are left in full.
- Add a section above the list of changes that briefly outlines any key changes in the release.
Post to the framework category with a subject matching the GitHub release title.
The post should resemble this:
The Charm Tech team has just released version x.y.z of ops!
It’s available from PyPI by using `pip install ops`, and `pip install ops[testing]`,
which will pick up the latest version. Upgrade by running `pip install --upgrade ops`.
The main improvements in this release are ...
Read more in the [full release notes on GitHub](link to the GitHub release).
In the post, outline the key improvements both in ops
and ops-scenario
-
the point here is to encourage people to check out the full notes and to upgrade
promptly, so ensure that you entice them with the best that the new versions
have to offer.