2016-12-16 15:48:06 -05:00
|
|
|
# All Rights Reserved.
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
|
|
# not use this file except in compliance with the License. You may obtain
|
|
|
|
# a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
# License for the specific language governing permissions and limitations
|
|
|
|
# under the License.
|
|
|
|
|
2018-02-08 15:22:02 -05:00
|
|
|
import json
|
2018-01-29 10:07:01 -05:00
|
|
|
import logging
|
2016-12-16 15:48:06 -05:00
|
|
|
import os
|
|
|
|
import os.path
|
2018-02-08 14:57:46 -05:00
|
|
|
import xmlrpc.client
|
2016-12-16 15:48:06 -05:00
|
|
|
|
2018-02-20 10:38:26 -05:00
|
|
|
from packaging import utils as packaging_utils
|
2018-01-29 10:07:01 -05:00
|
|
|
import requests
|
|
|
|
|
2018-02-07 12:01:35 -05:00
|
|
|
from openstack_releases import processutils
|
|
|
|
|
2018-01-29 10:07:01 -05:00
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
2016-12-16 15:48:06 -05:00
|
|
|
|
|
|
|
def get_sdist_name(workdir, repo):
|
2018-08-09 14:29:04 -04:00
|
|
|
"Find the name of the sdist."
|
2016-12-16 15:48:06 -05:00
|
|
|
dest = os.path.join(workdir, repo)
|
2018-03-27 14:51:03 -04:00
|
|
|
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)
|
2016-12-16 15:48:06 -05:00
|
|
|
return None
|
2017-02-08 07:52:56 -05:00
|
|
|
use_tox = repo.endswith('/pbr')
|
|
|
|
if use_tox and not os.path.exists(os.path.join(dest, '.tox', 'venv')):
|
|
|
|
# Use tox to set up a virtualenv so we can install the
|
|
|
|
# dependencies for the package. This only seems to be
|
|
|
|
# necessary for pbr, but...
|
2018-02-07 12:01:35 -05:00
|
|
|
processutils.check_output(
|
2017-02-08 07:52:56 -05:00
|
|
|
['tox', '-e', 'venv', '--notest'],
|
|
|
|
cwd=dest,
|
|
|
|
)
|
|
|
|
if use_tox:
|
|
|
|
python = '.tox/venv/bin/python'
|
|
|
|
else:
|
|
|
|
python = 'python'
|
2016-12-16 15:48:06 -05:00
|
|
|
# Run it once and discard the result to ensure any setup_requires
|
|
|
|
# dependencies are installed.
|
2017-02-08 07:52:56 -05:00
|
|
|
cmd = [python, 'setup.py', '--name']
|
2018-02-07 12:01:35 -05:00
|
|
|
processutils.check_output(cmd, cwd=dest)
|
2016-12-16 15:48:06 -05:00
|
|
|
# Run it again to get a clean version of the name.
|
2018-02-22 12:13:13 -05:00
|
|
|
LOG.debug('Running: %s in %s' % (' '.join(cmd), dest))
|
2018-02-07 12:01:35 -05:00
|
|
|
out = processutils.check_output(cmd, cwd=dest).decode('utf-8')
|
2018-02-22 12:13:13 -05:00
|
|
|
LOG.debug('Results: %s' % (out,))
|
2017-03-02 13:40:41 -05:00
|
|
|
name = out.splitlines()[-1].strip()
|
2016-12-16 15:48:06 -05:00
|
|
|
return name
|
2018-01-29 10:07:01 -05:00
|
|
|
|
|
|
|
|
2018-08-09 14:29:04 -04:00
|
|
|
def build_sdist(workdir, repo):
|
|
|
|
"Build the sdist."
|
|
|
|
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
|
|
|
|
use_tox = repo.endswith('/pbr')
|
|
|
|
if use_tox and not os.path.exists(os.path.join(dest, '.tox', 'venv')):
|
|
|
|
# Use tox to set up a virtualenv so we can install the
|
|
|
|
# dependencies for the package. This only seems to be
|
|
|
|
# necessary for pbr, but...
|
|
|
|
processutils.check_output(
|
|
|
|
['tox', '-e', 'venv', '--notest'],
|
|
|
|
cwd=dest,
|
|
|
|
)
|
|
|
|
if use_tox:
|
|
|
|
python = '.tox/venv/bin/python'
|
|
|
|
else:
|
|
|
|
python = 'python'
|
|
|
|
# Run it once and discard the result to ensure any setup_requires
|
|
|
|
# dependencies are installed.
|
|
|
|
cmd = [python, 'setup.py', 'sdist', 'bdist_wheel']
|
|
|
|
processutils.check_call(cmd, cwd=dest)
|
|
|
|
|
|
|
|
|
2018-05-29 14:40:46 -04:00
|
|
|
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,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-01-29 10:07:01 -05:00
|
|
|
def get_pypi_info(dist_name):
|
|
|
|
"Return PyPI information for the distribution."
|
2018-02-20 10:38:26 -05:00
|
|
|
canonical_name = packaging_utils.canonicalize_name(dist_name)
|
|
|
|
LOG.debug('looking at PyPI for {!r}'.format(canonical_name))
|
2018-04-20 17:29:00 +07:00
|
|
|
url = 'https://pypi.org/project/{}/json'.format(canonical_name)
|
2018-01-29 10:07:01 -05:00
|
|
|
LOG.debug(url)
|
2018-02-08 15:22:02 -05:00
|
|
|
try:
|
|
|
|
return requests.get(url).json()
|
|
|
|
except json.decoder.JSONDecodeError:
|
|
|
|
return {}
|
2018-02-08 14:57:46 -05:00
|
|
|
|
|
|
|
|
2018-02-20 10:38:26 -05:00
|
|
|
def _get_pypi_roles(dist_name):
|
2018-04-25 14:26:26 +02:00
|
|
|
client = xmlrpc.client.ServerProxy('https://pypi.org/pypi')
|
2018-02-20 10:38:26 -05:00
|
|
|
LOG.debug('retrieving roles for {!r}'.format(
|
|
|
|
dist_name))
|
|
|
|
return client.package_roles(dist_name)
|
|
|
|
|
|
|
|
|
|
|
|
def get_pypi_uploaders(dist_name):
|
|
|
|
roles = _get_pypi_roles(dist_name)
|
|
|
|
if not roles:
|
|
|
|
canonical_name = packaging_utils.canonicalize_name(dist_name)
|
|
|
|
roles = _get_pypi_roles(canonical_name)
|
2018-02-08 14:57:46 -05:00
|
|
|
uploaders = set(
|
|
|
|
acct
|
|
|
|
for role, acct in roles
|
|
|
|
if role in ('Owner', 'Maintainer')
|
|
|
|
)
|
2018-02-20 10:38:26 -05:00
|
|
|
LOG.debug('found: {}'.format(sorted(uploaders)))
|
2018-02-08 14:57:46 -05:00
|
|
|
return uploaders
|