Implement tasks to improve release automation

This commit is contained in:
Bruno Oliveira 2017-05-15 20:23:04 -03:00
parent 00e7ee532e
commit feab3ba70f
3 changed files with 84 additions and 24 deletions

View File

@ -1,27 +1,28 @@
How to release pytest How to release pytest
-------------------------------------------- --------------------------------------------
Note: this assumes you have already registered on PyPI and you have .. important::
`invoke <https://pypi.org/project/invoke/>`_ installed.
#. Check and finalize ``CHANGELOG.rst``. pytest releases must be prepared on **linux** because the docs and examples expect
to be executed in that platform.
#. Generate a new release announcement:: #. Install development dependencies in a virtual environment with::
invoke generate.announce VERSION pip3 install -r tasks/requirements.txt
Feel free to modify the generated files before committing. #. Create a branch ``release-X.Y.Z`` with the version for the release. Make sure it is up to date
with the latest ``master`` (for patch releases) and with the latest ``features`` merged with
the latest ``master`` (for minor releases). Ensure your are in a clean work tree.
#. Regenerate the docs examples using tox:: #. Check and finalize ``CHANGELOG.rst`` (will be automated soon).
tox -e regen #. Execute to automatically generate docs, announcements and upload a package to
your ``devpi`` staging server::
#. At this point, open a PR named ``release-X`` so others can help find regressions or provide suggestions. invoke generate.pre_release <VERSION> <DEVPI USER> --password <DEVPI PASSWORD>
#. Use devpi for uploading a release tarball to a staging area:: If ``--password`` is not given, it is assumed the user is already logged in. If you don't have
an account, please ask for one!
devpi use https://devpi.net/USER/dev
devpi upload --formats sdist,bdist_wheel
#. Run from multiple machines:: #. Run from multiple machines::

View File

@ -1,5 +1,6 @@
import os
from pathlib import Path from pathlib import Path
from subprocess import check_output from subprocess import check_output, check_call
import invoke import invoke
@ -9,11 +10,7 @@ import invoke
}) })
def announce(ctx, version): def announce(ctx, version):
"""Generates a new release announcement entry in the docs.""" """Generates a new release announcement entry in the docs."""
print("[generate.announce] Generating Announce")
# Get our list of authors # Get our list of authors
print("[generate.announce] Collecting author names")
stdout = check_output(["git", "describe", "--abbrev=0", '--tags']) stdout = check_output(["git", "describe", "--abbrev=0", '--tags'])
stdout = stdout.decode('utf-8') stdout = stdout.decode('utf-8')
last_version = stdout.strip() last_version = stdout.strip()
@ -29,12 +26,12 @@ def announce(ctx, version):
contributors_text = '\n'.join('* {}'.format(name) for name in sorted(contributors)) + '\n' contributors_text = '\n'.join('* {}'.format(name) for name in sorted(contributors)) + '\n'
text = template_text.format(version=version, contributors=contributors_text) text = template_text.format(version=version, contributors=contributors_text)
target = Path(__file__).joinpath('../../doc/en/announce/release-{}.rst'.format(version)) target = Path(__file__).parent.joinpath('../doc/en/announce/release-{}.rst'.format(version))
target.write_text(text, encoding='UTF-8') target.write_text(text, encoding='UTF-8')
print("[generate.announce] Generated {}".format(target.name)) print("[generate.announce] Generated {}".format(target.name))
# Update index with the new release entry # Update index with the new release entry
index_path = Path(__file__).joinpath('../../doc/en/announce/index.rst') index_path = Path(__file__).parent.joinpath('../doc/en/announce/index.rst')
lines = index_path.read_text(encoding='UTF-8').splitlines() lines = index_path.read_text(encoding='UTF-8').splitlines()
indent = ' ' indent = ' '
for index, line in enumerate(lines): for index, line in enumerate(lines):
@ -48,9 +45,68 @@ def announce(ctx, version):
print("[generate.announce] Skip {} (already contains release)".format(index_path.name)) print("[generate.announce] Skip {} (already contains release)".format(index_path.name))
break break
check_call(['git', 'add', str(target)])
@invoke.task()
def regen(ctx):
"""Call regendoc tool to update examples and pytest output in the docs."""
print("[generate.regen] Updating docs")
check_call(['tox', '-e', 'regen'])
@invoke.task()
def make_tag(ctx, version):
"""Create a new (local) tag for the release, only if the repository is clean."""
from git import Repo
repo = Repo('.')
if repo.is_dirty():
print('Current repository is dirty. Please commit any changes and try again.')
raise invoke.Exit(code=2)
tag_names = [x.name for x in repo.tags]
if version in tag_names:
print("[generate.make_tag] Delete existing tag {}".format(version))
repo.delete_tag(version)
print("[generate.make_tag] Create tag {}".format(version))
repo.create_tag(version)
@invoke.task()
def devpi_upload(ctx, version, user, password=None):
"""Creates and uploads a package to devpi for testing."""
if password:
print("[generate.devpi_upload] devpi login {}".format(user))
check_call(['devpi', 'login', user, '--password', password])
check_call(['devpi', 'use', 'https://devpi.net/{}/dev'.format(user)])
env = os.environ.copy()
env['SETUPTOOLS_SCM_PRETEND_VERSION'] = version
check_call(['devpi', 'upload', '--formats', 'sdist,bdist_wheel'], env=env)
print("[generate.devpi_upload] package uploaded")
@invoke.task(help={
'version': 'version being released',
'user': 'name of the user on devpi to stage the generated package',
'password': 'user password on devpi to stage the generated package '
'(if not given assumed logged in)',
})
def pre_release(ctx, version, user, password=None):
"""Generates new docs, release announcements and uploads a new release to devpi for testing."""
announce(ctx, version)
regen(ctx)
msg = 'Preparing release version {}'.format(version)
check_call(['git', 'commit', '-a', '-m', msg])
make_tag(ctx, version)
devpi_upload(ctx, version=version, user=user, password=password)
print() print()
print('Please review the generated files and commit with:') print('[generate.pre_release] Please push your branch and open a PR.')
print(' git commit -a -m "Generate new release announcement for {}'.format(version))

3
tasks/requirements.txt Normal file
View File

@ -0,0 +1,3 @@
invoke
tox
gitpython