Implement publishing to PyPI by pushing a tag

Fix #3060
This commit is contained in:
Bruno Oliveira 2018-02-09 20:59:15 -02:00
parent 00d3001138
commit bfe2cbe875
5 changed files with 37 additions and 96 deletions

View File

@ -2,10 +2,8 @@ sudo: false
language: python language: python
python: python:
- '3.6' - '3.6'
# command to install dependencies
install: install:
- pip install --upgrade --pre tox - pip install --upgrade --pre tox
# # command to run tests
env: env:
matrix: matrix:
# coveralls is not listed in tox's envlist, but should run in travis # coveralls is not listed in tox's envlist, but should run in travis
@ -29,7 +27,7 @@ env:
- TOXENV=doctesting - TOXENV=doctesting
- TOXENV=docs - TOXENV=docs
matrix: jobs:
include: include:
- env: TOXENV=pypy - env: TOXENV=pypy
python: 'pypy-5.4' python: 'pypy-5.4'
@ -40,6 +38,22 @@ matrix:
- env: TOXENV=py37 - env: TOXENV=py37
python: 'nightly' python: 'nightly'
- stage: deploy
python: '3.6'
env:
install: pip install -U setuptools setuptools_scm
script: skip
deploy:
provider: pypi
user: nicoddemus
distributions: sdist bdist_wheel
skip_upload_docs: true
password:
secure: xanTgTUu6XDQVqB/0bwJQXoDMnU5tkwZc5koz6mBkkqZhKdNOi2CLoC1XhiSZ+ah24l4V1E0GAqY5kBBcy9d7NVe4WNg4tD095LsHw+CRU6/HCVIFfyk2IZ+FPAlguesCcUiJSXOrlBF+Wj68wEvLoK7EoRFbJeiZ/f91Ww1sbtDlqXABWGHrmhPJL5Wva7o7+wG7JwJowqdZg1pbQExsCc7b53w4v2RBu3D6TJaTAzHiVsW+nUSI67vKI/uf+cR/OixsTfy37wlHgSwihYmrYLFls3V0bSpahCim3bCgMaFZx8S8xrdgJ++PzBCof2HeflFKvW+VCkoYzGEG4NrTWJoNz6ni4red9GdvfjGH3YCjAKS56h9x58zp2E5rpsb/kVq5/45xzV+dq6JRuhQ1nJWjBC6fSKAc/bfwnuFK3EBxNLkvBssLHvsNjj5XG++cB8DdS9wVGUqjpoK4puaXUWFqy4q3S9F86HEsKNgExtieA9qNx+pCIZVs6JCXZNjr0I5eVNzqJIyggNgJG6RyravsU35t9Zd9doL5g4Y7UKmAGTn1Sz24HQ4sMQgXdm2SyD8gEK5je4tlhUvfGtDvMSlstq71kIn9nRpFnqB6MFlbYSEAZmo8dGbCquoUc++6Rum208wcVbrzzVtGlXB/Ow9AbFMYeAGA0+N/K1e59c=
on:
tags: true
repo: pytest-dev/pytest
script: tox --recreate script: tox --recreate
notifications: notifications:

View File

@ -22,44 +22,28 @@ taking a lot of time to make a new one.
Ensure your are in a clean work tree. Ensure your are in a clean work tree.
#. Generate docs, changelog, announcements and upload a package to #. Generate docs, changelog, announcements and a **local** tag::
your ``devpi`` staging server::
invoke generate.pre-release <VERSION> <DEVPI USER> --password <DEVPI PASSWORD> invoke generate.pre-release <VERSION>
If ``--password`` is not given, it is assumed the user is already logged in ``devpi``.
If you don't have an account, please ask for one.
#. Open a PR for this branch targeting ``master``. #. Open a PR for this branch targeting ``master``.
#. Test the package #. After all tests pass and the PR has been approved, publish to PyPI by pushing the tag::
* **Manual method** git push git@github.com:pytest-dev/pytest.git <VERSION>
Run from multiple machines:: Wait for the deploy to complete, then make sure it is `available on PyPI <https://pypi.org/project/pytest>`_.
devpi use https://devpi.net/USER/dev #. Send an email announcement with the contents from::
devpi test pytest==VERSION
Check that tests pass for relevant combinations with:: doc/en/announce/release-<VERSION>.rst
devpi list pytest To the following mailing lists:
* **CI servers** * pytest-dev@python.org (all releases)
* python-announce-list@python.org (all releases)
* testing-in-python@lists.idyll.org (only major/minor releases)
Configure a repository as per-instructions on And announce it on `Twitter <https://twitter.com/>`_ with the ``#pytest`` hashtag.
devpi-cloud-test_ to test the package on Travis_ and AppVeyor_.
All test environments should pass.
#. Publish to PyPI::
invoke generate.publish-release <VERSION> <DEVPI USER> <PYPI_NAME>
where PYPI_NAME is the name of pypi.python.org as configured in your ``~/.pypirc``
file `for devpi <http://doc.devpi.net/latest/quickstart-releaseprocess.html?highlight=pypirc#devpi-push-releasing-to-an-external-index>`_.
#. After a minor/major release, merge ``release-X.Y.Z`` into ``master`` and push (or open a PR). #. After a minor/major release, merge ``release-X.Y.Z`` into ``master`` and push (or open a PR).
.. _devpi-cloud-test: https://github.com/obestwalter/devpi-cloud-test
.. _AppVeyor: https://www.appveyor.com/
.. _Travis: https://travis-ci.org

View File

@ -0,0 +1 @@
pytest has changed the publication procedure and is now being published to PyPI directly from Travis.

View File

@ -1,4 +1,6 @@
import os """
Invoke development tasks.
"""
from pathlib import Path from pathlib import Path
from subprocess import check_output, check_call from subprocess import check_output, check_call
@ -57,7 +59,7 @@ def regen(ctx):
@invoke.task() @invoke.task()
def make_tag(ctx, version): def make_tag(ctx, version):
"""Create a new (local) tag for the release, only if the repository is clean.""" """Create a new, local tag for the release, only if the repository is clean."""
from git import Repo from git import Repo
repo = Repo('.') repo = Repo('.')
@ -74,81 +76,24 @@ def make_tag(ctx, version):
repo.create_tag(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={ @invoke.task(help={
'version': 'version being released', '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): def pre_release(ctx, version):
"""Generates new docs, release announcements and uploads a new release to devpi for testing.""" """Generates new docs, release announcements and creates a local tag."""
announce(ctx, version) announce(ctx, version)
regen(ctx) regen(ctx)
changelog(ctx, version, write_out=True) changelog(ctx, version, write_out=True)
msg = 'Preparing release version {}'.format(version) msg = 'Preparing release version {}'.format(version)
check_call(['git', 'commit', '-a', '-m', msg]) check_call(['git', 'commit', '-a', '-m', msg])
make_tag(ctx, version) make_tag(ctx, version)
devpi_upload(ctx, version=version, user=user, password=password)
print() print()
print('[generate.pre_release] Please push your branch and open a PR.') print('[generate.pre_release] Please push your branch and open a PR.')
@invoke.task(help={
'version': 'version being released',
'user': 'name of the user on devpi to stage the generated package',
'pypi_name': 'name of the pypi configuration section in your ~/.pypirc',
})
def publish_release(ctx, version, user, pypi_name):
"""Publishes a package previously created by the 'pre_release' command."""
from git import Repo
repo = Repo('.')
tag_names = [x.name for x in repo.tags]
if version not in tag_names:
print('Could not find tag for version {}, exiting...'.format(version))
raise invoke.Exit(code=2)
check_call(['devpi', 'use', 'https://devpi.net/{}/dev'.format(user)])
check_call(['devpi', 'push', 'pytest=={}'.format(version), 'pypi:{}'.format(pypi_name)])
check_call(['git', 'push', 'git@github.com:pytest-dev/pytest.git', version])
emails = [
'pytest-dev@python.org',
'python-announce-list@python.org'
]
if version.endswith('.0'):
emails.append('testing-in-python@lists.idyll.org')
print('Version {} has been published to PyPI!'.format(version))
print()
print('Please send an email announcement with the contents from:')
print()
print(' doc/en/announce/release-{}.rst'.format(version))
print()
print('To the following mail lists:')
print()
print(' ', ','.join(emails))
print()
print('And announce it on twitter adding the #pytest hash tag.')
@invoke.task(help={ @invoke.task(help={
'version': 'version being released', 'version': 'version being released',
'write_out': 'write changes to the actual changelog' 'write_out': 'write changes to the actual changelog'
@ -159,4 +104,3 @@ def changelog(ctx, version, write_out=False):
else: else:
addopts = ['--draft'] addopts = ['--draft']
check_call(['towncrier', '--version', version] + addopts) check_call(['towncrier', '--version', version] + addopts)

View File

@ -1,6 +1,4 @@
devpi-client
gitpython gitpython
invoke invoke
towncrier towncrier
tox tox
wheel