add a validation step to check README.rst parsing for PyPI
PyPI is now more strict about requiring README.rst files to be parseable. Failures to parse the readme block uploading new versions of packages. To detect these errors in advance, we should validate the README formatting as part of validating a release. Change-Id: I416befcf3e47818680018394fa63de8983351587 Signed-off-by: Doug Hellmann <doug@doughellmann.com>
This commit is contained in:
parent
5afa036729
commit
dcb0bbaa27
openstack_releases
@ -570,6 +570,49 @@ def validate_tarball_base(deliv, context):
|
|||||||
sdist, expected))
|
sdist, expected))
|
||||||
|
|
||||||
|
|
||||||
|
@applies_to_released
|
||||||
|
def validate_pypi_readme(deliv, context):
|
||||||
|
"Does the README look right for PyPI?"
|
||||||
|
|
||||||
|
# Check out the repositories to the hash for the latest release on
|
||||||
|
# the branch so we can find the setup.py and README.rst there (if
|
||||||
|
# it exists), in case that has been removed from master after a
|
||||||
|
# project is retired. This also ensures we get the right name for
|
||||||
|
# the branch, in case the sdist name changes over time.
|
||||||
|
latest_release = deliv.releases[-1]
|
||||||
|
for project in latest_release.projects:
|
||||||
|
gitutils.safe_clone_repo(
|
||||||
|
context.workdir, project.repo.name, project.hash, context)
|
||||||
|
|
||||||
|
for repo in deliv.repos:
|
||||||
|
|
||||||
|
job_templates = context.zuul_projects.get(repo.name, {}).get(
|
||||||
|
'templates', [])
|
||||||
|
LOG.debug('{} has job templates {}'.format(repo.name, job_templates))
|
||||||
|
|
||||||
|
# Look for jobs that appear to be talking about publishing to
|
||||||
|
# PyPI. There are variations.
|
||||||
|
pypi_jobs = [
|
||||||
|
j
|
||||||
|
for j in job_templates
|
||||||
|
if 'pypi' in j
|
||||||
|
]
|
||||||
|
|
||||||
|
if not pypi_jobs:
|
||||||
|
print('rule only applies to repos publishing to PyPI')
|
||||||
|
continue
|
||||||
|
|
||||||
|
LOG.debug('{} publishes to PyPI via {}'.format(repo.name, pypi_jobs))
|
||||||
|
|
||||||
|
try:
|
||||||
|
pythonutils.check_readme_format(context.workdir, repo.name)
|
||||||
|
except Exception as err:
|
||||||
|
context.error('README check for {} failed: {}'.format(
|
||||||
|
repo.name, err))
|
||||||
|
else:
|
||||||
|
print('OK')
|
||||||
|
|
||||||
|
|
||||||
@applies_to_released
|
@applies_to_released
|
||||||
def validate_pypi_permissions(deliv, context):
|
def validate_pypi_permissions(deliv, context):
|
||||||
"Do we have permission to upload to PyPI?"
|
"Do we have permission to upload to PyPI?"
|
||||||
@ -1511,6 +1554,7 @@ def main():
|
|||||||
validate_model,
|
validate_model,
|
||||||
validate_release_type,
|
validate_release_type,
|
||||||
validate_pypi_permissions,
|
validate_pypi_permissions,
|
||||||
|
validate_pypi_readme,
|
||||||
validate_gitreview,
|
validate_gitreview,
|
||||||
validate_release_sha_exists,
|
validate_release_sha_exists,
|
||||||
validate_existing_tags,
|
validate_existing_tags,
|
||||||
|
@ -59,6 +59,22 @@ def get_sdist_name(workdir, repo):
|
|||||||
return name
|
return name
|
||||||
|
|
||||||
|
|
||||||
|
def check_readme_format(workdir, repo):
|
||||||
|
"Verify that the README format looks OK."
|
||||||
|
dest = os.path.join(workdir, repo)
|
||||||
|
setup_path = os.path.join(dest, 'setup.py')
|
||||||
|
if not os.path.exists(setup_path):
|
||||||
|
LOG.debug('did not find %s, maybe %s is not a python project',
|
||||||
|
setup_path, repo)
|
||||||
|
return None
|
||||||
|
# NOTE(dhellmann): This relies on validate being run via tox so
|
||||||
|
# that python3 is present and the docutils package is installed.
|
||||||
|
processutils.check_call(
|
||||||
|
['python3', 'setup.py', 'check', '--restructuredtext', '--strict'],
|
||||||
|
cwd=dest,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def get_pypi_info(dist_name):
|
def get_pypi_info(dist_name):
|
||||||
"Return PyPI information for the distribution."
|
"Return PyPI information for the distribution."
|
||||||
canonical_name = packaging_utils.canonicalize_name(dist_name)
|
canonical_name = packaging_utils.canonicalize_name(dist_name)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user