Skip to content

Commit

Permalink
Switch from Reno to Towncrier for release notes (Qiskit#1501)
Browse files Browse the repository at this point in the history
## Why Reno broke

Closes Qiskit/documentation#979 and Qiskit/documentation#978.

Reno stopped working correctly with Runtime 0.21. The release notes for 0.16+ were all being included in the 0.15 release note entry, incorrectly. Qiskit#1486 attempted to fix this and it worked locally, but in CI the `stable/0.15` entry was missing all release notes before 0.15 and we could not figure out how to fix that.

Reno got into a bad state because of the feature branch `experimental-0.2`. That branch was based off of `main` at the time of 0.15.0. It was not merged back into `main` until preparing 0.21.0 when we merged 32b0dbc into `main` . During the feature branch's life, ec8f5a4 merged main into experimental-0.2, which we think messed things up by copying over Git tags into the feature branch. `git log --tags --simplify-by-decoration` shows all the tags in order, but between 0.20.0 and 0.21.0 is that commit ec8f5a4.

We think the problem is that Reno gets confused when you merge a branch that itself has Git tags back into `main`. This [blog warns](https://medium.com/opsops/software-discovery-reno-b072827ad883)

> There is one case when reno may skip release note, and it’s a single known ‘don’t do it’ case, documented and articulated: Do not merge stable branch with release tags back into master branch.

We think Reno ended up confusing some of the same Git tags between the feature branch and main.

We could not think of a safe fix for Reno because we cannot safely force push changes to the Git history. We couldn't find a way to teach Reno which release notes correspond to which version because it tries automating this all via Git.

## Switching to Towncrier

[Towncrier](https://towncrier.readthedocs.io/en/stable/) is a popular release notes generator in the Python community used by projects like pip and pytest. 

It is much simpler than Reno because it is less magical: it doesn't even look at Git. Instead, Towncrier has a simple workflow:

- Until a release, each change has a dedicated release note file describing it.
- When preparing a release note, you manually run `towncrier build` and it will combine all the separate release notes together and add them to an aggregated release note file for the entire release, like `0.20.0.rst`. It will also delete the individual release note files since they are now moved into the aggregated file.

Unlike Reno, this means that you only have standalone release notes files for unreleased versions. If you want to make changes to prior release notes, you simplify modify the aggregated file for that release, like changing `0.20.0.rst`.

This means there is a new step to preparing releases: you have to first create a PR to prep the release notes.

### Why do we have one release note file per release?

By default, towncrier writes every release note to a single file like `CHANGES.rst`. We instead have one file per release, like `0.20.0.rst`. We do this because it makes patch releases like 0.20.1 simpler; when we do a patch release, we need to cherry-pick the release prep from the `stable/x` branch back into `main` so that the release note file for 0.20.1 shows up on `main`. By having the release notes be separated files per release, we avoid that cherry-pick from having a merge conflict with `main`.

We have a helper script that Tox and Make call to concatenate all these separate release note files into a single `release_notes.rst` to be used by the Sphin build.
  • Loading branch information
Eric-Arellano authored and kt474 committed Mar 18, 2024
1 parent 6e758c8 commit ebe154d
Show file tree
Hide file tree
Showing 225 changed files with 1,786 additions and 1,575 deletions.
2 changes: 0 additions & 2 deletions .github/workflows/docs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,6 @@ jobs:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Set up Python
uses: actions/setup-python@v5
with:
Expand Down
3 changes: 3 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -110,3 +110,6 @@ Qconfig.py

# Pycharm
.idea

# Generated release notes file
docs/release_notes.rst
209 changes: 94 additions & 115 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -61,140 +61,103 @@ The current categories for each label are as follows:

### Release Notes

When making any end user facing changes in a contribution we have to make sure
we document that when we release a new version of qiskit-ibm. The
expectation is that if your code contribution has user facing changes that you
When making any end user facing changes in a contribution, we have to make sure
we document that when we release a new version of qiskit-ibm-runtime. The
expectation is that if your code contribution has user facing changes, then you
will write the release documentation for these changes. This documentation must
explain what was changed, why it was changed, and how users can either use or
adapt to the change. The idea behind release documentation is that when a naive
user with limited internal knowledge of the project is upgrading from the
previous release to the new one, they should be able to read the release notes,
understand if they need to update their program which uses qiskit, and how they
would go about doing that. It ideally should explain why they need to make
this change too, to provide the necessary context.
understand if they need to update their program which uses qiskit-ibm-runtime,
and how they would go about doing that. It ideally should explain why
they need to make this change too, to provide the necessary context.

To make sure we don't forget a release note or if the details of user facing
changes over a release cycle we require that all user facing changes include
documentation at the same time as the code. To accomplish this we use the
[reno](https://docs.openstack.org/reno/latest/) tool which enables a git based
workflow for writing and compiling release notes.
documentation at the same time as the code. To accomplish this, we use the
[Towncrier](https://towncrier.readthedocs.io/en/stable/) tool.

#### Adding a new release note

Making a new release note is quite straightforward. Ensure that you have reno
installed with::

pip install -U reno

Once you have reno installed you can make a new release note by running in
your local repository checkout's root::

reno new short-description-string

where short-description-string is a brief string (with no spaces) that describes
what's in the release note. This will become the prefix for the release note
file. Once that is run it will create a new yaml file in releasenotes/notes.
Then open that yaml file in a text editor and write the release note. The basic
structure of a release note is restructured text in yaml lists under category
keys. You add individual items under each category and they will be grouped
automatically by release when the release notes are compiled. A single file
can have as many entries in it as needed, but to avoid potential conflicts
you'll want to create a new file for each pull request that has user facing
changes. When you open the newly created file it will be a full template of
the different categories with a description of a category as a single entry
in each category. You'll want to delete all the sections you aren't using and
update the contents for those you are. For example, the end result should
look something like::

```yaml
features:
- |
Introduced a new feature foo, that adds support for doing something to
``QuantumCircuit`` objects. It can be used by using the foo function,
for example::
from qiskit import foo
from qiskit import QuantumCircuit
foo(QuantumCircuit())
- |
The ``qiskit.QuantumCircuit`` module has a new method ``foo()``. This is
the equivalent of calling the ``qiskit.foo()`` to do something to your
QuantumCircuit. This is the equivalent of running ``qiskit.foo()`` on
your circuit, but provides the convenience of running it natively on
an object. For example::
from qiskit import QuantumCircuit
circ = QuantumCircuit()
circ.foo()
deprecations:
- |
The ``qiskit.bar`` module has been deprecated and will be removed in a
future release. Its sole function, ``foobar()`` has been superseded by the
``qiskit.foo()`` function which provides similar functionality but with
more accurate results and better performance. You should update your calls
``qiskit.bar.foobar()`` calls to ``qiskit.foo()``.
```
To create a new release note, first find either the issue or PR number associated with
your change from GitHub because Towncrier links every release note to a GitHub issue
or PR. If there is no associated issue and you haven't yet opened up the PR so you don't
yet have a PR number, you can use the value `todo` at first, then go back and rename the
file once you open up the PR and have its number.

You can also look at other release notes for other examples.
Then, identify which type of change your release note is:

You can use any restructured text feature in them (code sections, tables,
enumerated lists, bulleted list, etc) to express what is being changed as
needed. In general you want the release notes to include as much detail as
needed so that users will understand what has changed, why it changed, and how
they'll have to update their code.
- `feat` (new feature)
- `upgrade` (upgrade note)
- `deprecation` (deprecation)
- `bug` (bug fix)
- `other` (other note)

After you've finished writing your release notes you'll want to add the note
file to your commit with `git add` and commit them to your PR branch to make
sure they're included with the code in your PR.
Now, create a new file in the `release-notes/unreleased` folder in the format `<github-number>.<type>.rst`,
such as `156.bug.rst` or `231.feat.rst`.

#### Linking to issues
Open up the new release note file and provide a description of the change, such as what users need
to do. The files use RST syntax and you can use mechanisms like code blocks and cross-references.

If you need to link to an issue or other github artifact as part of the release
note this should be done using an inline link with the text being the issue
number. For example you would write a release note with a link to issue 12345
as:
Example notes:

```yaml
fixes:
- |
Fixes a race condition in the function ``foo()``. Refer to
`#12345 <https://github.com/Qiskit/qiskit-ibm-runtime/issues/12345>`_ for
more details.
```rst
Add ``dd_barrier`` optional input to
:class:`.PadDynamicalDecoupling`
constructor to identify portions of the circuit to apply dynamical
decoupling (dd) on selectively. If this string is contained in the
label of a barrier in the circuit, dd is applied on the delays ending
with it (on the same qubits); otherwise, it is not applied.
```

#### Generating the release notes
```
When a single backend is retrieved with the ``instance`` parameter,
After release notes have been added if you want to see what the full output of
the release notes. In general the output from reno that we'll get is a rst
(ReStructuredText) file that can be compiled by
[sphinx](https://www.sphinx-doc.org/en/master/). To generate the rst file you
use the ``reno report`` command. If you want to generate the full Qiskit IBM provider
release notes for all releases (since we started using reno during 0.9) you just
run::
.. code:: python
reno report
service.backend('ibm_torino', instance='ibm-q/open/main')
# raises error if torino is not in ibm-q/open/main but in a different instance
# the user has access to
service = QiskitRuntimeService(channel="ibm_quantum", instance="ibm-q/open/main")
service.backend('ibm_torino') # raises the same error
but you can also use the ``--version`` argument to view a single release (after
it has been tagged::
if the backend is not in the instance, but in a different one the user
has access to, an error will be raised. The same error will now be
raised if an instance is passed in at initialization and then a
backend not in that instance is retrieved.
```

reno report --version 0.9.0
In general, you want the release notes to include as much detail as
needed so that users will understand what has changed, why it changed, and how
they'll have to update their code.

At release time ``reno report`` is used to generate the release notes for the
release and the output will be submitted as a pull request to the documentation
repository's [release notes file](
https://github.com/Qiskit/qiskit/blob/master/docs/release_notes.rst)
Towncrier will automatically add a link to the PR or Issue number you used in
the file name once we build the relesae notes during the release.

After you've finished writing your release note, you need to add the note
file to your commit with `git add` and commit them to your PR branch to make
sure they're included with the code in your PR.

#### Preview the release notes

You can preview how the release notes look with the Sphinx docs build by
using Towncrier. First, install Towncrier with [`pipx`](https://pipx.pypa.io/stable/) by
running `pipx install tonwcrier`.

Then, run `towncrier build --version=unreleased --keep`. Be careful to not save the file `unreleased.rst` to Git!

Finally, preview the docs build by following the instructions in
[Building documentation locally](#building-documentation-locally).

### Building documentation locally

Building The release notes are part of the standard qiskit-ibm
Building The release notes are part of the standard qiskit-ibm-runtime
documentation builds. To check what the rendered html output of the release
notes will look like for the current state of the repo you can run:
`tox -edocs` which will build all the documentation into `docs/_build/html`
and the release notes in particular will be located at
`docs/_build/html/release_notes.html`
`docs/_build/html/release_notes.html`.

### Test

Expand Down Expand Up @@ -314,19 +277,35 @@ merged to it are bugfixes.

### Release cycle

When it is time to release a new minor version of qiskit-ibm we will:
When it is time to release a new minor version of qiskit-ibm-runtime, first open a PR
to prepare the release notes. Install the tool `towncrier` with `pipx install towncrier`.
Then, in a new branch, run `towncrier build --version=<full-version> --yes`, and replace
`<full-version>` with the version like `0.22.0`. Add all the changes to Git and
open a PR.

After landing the release notes preparation, checkout `main` and make sure that the last
commit is the release notes prep. Then, create a new Git tag from `main` for the full
version number, like `git tag 0.22.0`. Push the tag to GitHub. Also create a new branch like
`stable/0.22` and push it to GitHub.

GitHub Actions will automatically build and upload the wheels to PyPI. The
qiskit-bot should also automatically create the GitHub Release for you.

#### Patch releases

1. Create a new tag with the version number and push it to github
2. Change the `main` version to the next release version.
The `stable/*` branches should only receive changes in the form of bug fixes.
These bug fixes should first land on `main`, then be `git cherry-pick`ed to
the stable branch. Include the Towncrier release note in these cherry-picks.

The release automation processes will be triggered by the new tag and perform
the following steps:
When preparing a patch release, you also need to first land a PR against
the `stable/*` branch to prepare the release notes with
`towncrier build --version=<full-version> --yes`, where `<full-version>` is
the patch release like `0.21.1`. Then, from the `stable/*` branch, create a new
Git tag for the full version number, like `git tag 0.21.1`, and
push the tag to GitHub.

1. Create a stable branch for the new minor version from the release tag
on the `main` branch
2. Build and upload binary wheels to pypi
3. Create a github release page with a generated changelog
4. Generate a PR on the meta-repository to bump the Qiskit IBM Provider version
and meta-package version.
GitHub Actions will automatically build and upload the wheels to PyPI. The
qiskit-bot should also automatically create the GitHub Release for you.

The `stable/*` branches should only receive changes in the form of bug fixes.
Finally, you need to cherry-pick the release notes prep from `stable/*` to
the `main` branch, such as from `stable/0.21` to `main`.
1 change: 1 addition & 0 deletions docs/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -25,4 +25,5 @@ help:
# Catch-all target: route all unknown targets to Sphinx using the new
# "make mode" option. $(O) is meant as a shortcut for $(SPHINXOPTS).
%: Makefile
python ../tools/concat_release_notes.py
@$(SPHINXBUILD) -M $@ "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
1 change: 0 additions & 1 deletion docs/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,6 @@
"sphinx.ext.linkcode",
'jupyter_sphinx',
'sphinx_autodoc_typehints',
'reno.sphinxext',
'nbsphinx',
'sphinxcontrib.katex',
'matplotlib.sphinxext.plot_directive',
Expand Down
16 changes: 0 additions & 16 deletions docs/release_notes.rst

This file was deleted.

32 changes: 32 additions & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,35 @@ build-backend = "setuptools.build_meta"
[tool.black]
line-length = 100
target-versions = ['py38', 'py39', 'py310', 'py311', 'py312']

[tool.towncrier]
single_file = false
filename = "release-notes/{version}.rst"
directory = "release-notes/unreleased"
title_format = "{version} ({project_date})"
issue_format = "`{issue} <https://https://github.com/Qiskit/qiskit-ibm-runtime/pull/{issue}>`__"

[[tool.towncrier.type]]
directory = "upgrade"
name = "Upgrade Notes"
showcontent = true

[[tool.towncrier.type]]
directory = "deprecation"
name = "Deprecation Notes"
showcontent = true

[[tool.towncrier.type]]
directory = "feat"
name = "New Features"
showcontent = true

[[tool.towncrier.type]]
directory = "bug"
name = "Bug Fixes"
showcontent = true

[[tool.towncrier.type]]
directory = "other"
name = "Other Notes"
showcontent = true
Loading

0 comments on commit ebe154d

Please sign in to comment.