2015-07-23 18:17:57 +00: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.
|
|
|
|
|
|
|
|
"""Show the changes that will be included in the release.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import atexit
|
2015-10-06 13:47:39 +00:00
|
|
|
import glob
|
2017-08-04 13:58:15 +00:00
|
|
|
import json
|
2017-10-03 20:20:15 +00:00
|
|
|
import logging
|
2015-07-23 18:17:57 +00:00
|
|
|
import os
|
|
|
|
import os.path
|
[list_changes] Handle non-numeric tags
After we tagged a realease EM git describe on a branch will include that
tag and as it contains a '-' it breaks the partition() we're using to
get the tag[1].
Switch to using a re rather than the simple partition.
(venv) [tony@thor releases]$ ipython
Python 3.7.3 (default, Mar 27 2019, 13:41:07)
Type 'copyright', 'credits' or 'license' for more information
IPython 7.4.0 -- An enhanced Interactive Python. Type '?' for help.
In [1]: import openstack_releases.cmds.list_changes
In [2]: workdir = '/home/tony/projects/openstack'
In [3]: repo = 'openstack/python-manilaclient'
In [4]: openstack_releases.cmds.list_changes.git_list_existing_branches(workdir, repo)
All Branches with Version Numbers
---------------------------------
feature/add-constraints-support 1.11.0-36-g7134e3c 1.11.0 2 years, 8 months ago
master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
review/openstack_release_bot/create-stein 1.27.0-2-g31608cf 1.27.0 6 months ago
remotes/gerrit/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
fatal: ambiguous argument 'ocata': unknown revision or path not in the working tree.
Use '--' to separate paths from revisions, like this:
'git <command> [<revision>...] -- [<file>...]'
remotes/gerrit/stable/ocata ocata-em-5-g22e9c42 ocata
remotes/gerrit/stable/pike 1.17.4 1.17.4 7 months ago
remotes/gerrit/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/gerrit/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/gerrit/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
remotes/origin/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
fatal: ambiguous argument 'ocata': unknown revision or path not in the working tree.
Use '--' to separate paths from revisions, like this:
'git <command> [<revision>...] -- [<file>...]'
remotes/origin/stable/ocata ocata-em-5-g22e9c42 ocata
remotes/origin/stable/pike 1.17.4 1.17.4 7 months ago
remotes/origin/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/origin/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/origin/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
In [5]: from imp import reload
In [6]: reload(openstack_releases.cmds.list_changes)
Out[6]: <module 'openstack_releases.cmds.list_changes' from '/home/tony/projects/openstack/openstack/releases/openstack_releases/cmds/list_changes.py'>
In [7]: openstack_releases.cmds.list_changes.git_list_existing_branches(workdir, repo)
All Branches with Version Numbers
---------------------------------
feature/add-constraints-support 1.11.0-36-g7134e3c 1.11.0 2 years, 8 months ago
master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
review/openstack_release_bot/create-stein 1.27.0-2-g31608cf 1.27.0 6 months ago
remotes/gerrit/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
remotes/gerrit/stable/ocata ocata-em-5-g22e9c42 ocata-em 2 years, 3 months ago
remotes/gerrit/stable/pike 1.17.4
remotes/gerrit/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/gerrit/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/gerrit/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
remotes/origin/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
remotes/origin/stable/ocata ocata-em-5-g22e9c42 ocata-em 2 years, 3 months ago
remotes/origin/stable/pike 1.17.4
remotes/origin/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/origin/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/origin/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
[1] http://logs.openstack.org/75/652775/2/check/releases-tox-list-changes/92b219a/job-output.txt.gz#_2019-04-16_14_25_50_377843
Change-Id: I6a6cf9110a737204e332c60004f72a68b770d1e7
2019-04-17 01:15:34 +00:00
|
|
|
import re
|
2015-07-23 18:17:57 +00:00
|
|
|
import shutil
|
|
|
|
import subprocess
|
2016-09-29 16:59:56 +00:00
|
|
|
import sys
|
2015-07-23 18:17:57 +00:00
|
|
|
import tempfile
|
|
|
|
|
2018-10-12 20:05:11 +00:00
|
|
|
from openstack_governance import governance
|
2017-11-28 16:04:26 +00:00
|
|
|
import pyfiglet
|
2017-08-04 13:58:15 +00:00
|
|
|
import requests
|
|
|
|
|
2017-11-28 16:04:26 +00:00
|
|
|
from openstack_releases import defaults
|
2018-03-08 18:16:08 +00:00
|
|
|
from openstack_releases import deliverable
|
2015-07-23 18:17:57 +00:00
|
|
|
from openstack_releases import gitutils
|
2018-02-08 16:53:52 +00:00
|
|
|
from openstack_releases import hound
|
2019-11-14 23:21:32 +00:00
|
|
|
from openstack_releases import liaisons
|
2017-10-03 20:21:30 +00:00
|
|
|
from openstack_releases import release_notes
|
2017-08-01 21:17:29 +00:00
|
|
|
from openstack_releases import yamlutils
|
2015-07-23 18:17:57 +00:00
|
|
|
|
|
|
|
|
2015-10-06 13:47:39 +00:00
|
|
|
def header(title):
|
|
|
|
print('\n%s' % title)
|
|
|
|
print('-' * len(title))
|
|
|
|
|
|
|
|
|
2017-11-28 16:04:26 +00:00
|
|
|
def banner(text):
|
|
|
|
pyfiglet.print_figlet(text, font='banner', width=120)
|
|
|
|
|
|
|
|
|
2016-02-29 15:15:57 +00:00
|
|
|
def git_show(workdir, repo, title, ref):
|
|
|
|
header('%s %s' % (title, ref))
|
|
|
|
cmd = ['git', 'log', '-n', '1', '--decorate', '--format=medium', ref]
|
|
|
|
print('\n' + ' '.join(cmd) + '\n')
|
|
|
|
subprocess.check_call(cmd, cwd=os.path.join(workdir, repo))
|
|
|
|
print()
|
|
|
|
|
|
|
|
|
2015-10-01 19:40:15 +00:00
|
|
|
def git_log(workdir, repo, title, git_range, extra_args=[]):
|
2015-10-06 13:47:39 +00:00
|
|
|
header('%s %s' % (title, git_range))
|
2015-10-01 19:40:15 +00:00
|
|
|
cmd = ['git', 'log', '--no-color']
|
|
|
|
cmd.extend(extra_args)
|
2016-02-03 17:13:13 +00:00
|
|
|
if isinstance(git_range, str):
|
|
|
|
cmd.append(git_range)
|
|
|
|
else:
|
|
|
|
cmd.extend(git_range)
|
|
|
|
print('\n' + ' '.join(cmd) + '\n')
|
2015-10-01 19:40:15 +00:00
|
|
|
subprocess.check_call(cmd, cwd=os.path.join(workdir, repo))
|
2015-08-13 18:22:34 +00:00
|
|
|
print()
|
|
|
|
|
|
|
|
|
2017-02-16 17:10:43 +00:00
|
|
|
def git_list_existing_branches(workdir, repo):
|
|
|
|
header('All Branches with Version Numbers')
|
2017-01-31 20:01:30 +00:00
|
|
|
for branch in gitutils.get_branches(workdir, repo):
|
2017-03-20 15:17:19 +00:00
|
|
|
try:
|
|
|
|
description = subprocess.check_output(
|
|
|
|
['git', 'describe', branch],
|
|
|
|
cwd=os.path.join(workdir, repo),
|
|
|
|
).decode('utf-8').strip()
|
[list_changes] Handle non-numeric tags
After we tagged a realease EM git describe on a branch will include that
tag and as it contains a '-' it breaks the partition() we're using to
get the tag[1].
Switch to using a re rather than the simple partition.
(venv) [tony@thor releases]$ ipython
Python 3.7.3 (default, Mar 27 2019, 13:41:07)
Type 'copyright', 'credits' or 'license' for more information
IPython 7.4.0 -- An enhanced Interactive Python. Type '?' for help.
In [1]: import openstack_releases.cmds.list_changes
In [2]: workdir = '/home/tony/projects/openstack'
In [3]: repo = 'openstack/python-manilaclient'
In [4]: openstack_releases.cmds.list_changes.git_list_existing_branches(workdir, repo)
All Branches with Version Numbers
---------------------------------
feature/add-constraints-support 1.11.0-36-g7134e3c 1.11.0 2 years, 8 months ago
master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
review/openstack_release_bot/create-stein 1.27.0-2-g31608cf 1.27.0 6 months ago
remotes/gerrit/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
fatal: ambiguous argument 'ocata': unknown revision or path not in the working tree.
Use '--' to separate paths from revisions, like this:
'git <command> [<revision>...] -- [<file>...]'
remotes/gerrit/stable/ocata ocata-em-5-g22e9c42 ocata
remotes/gerrit/stable/pike 1.17.4 1.17.4 7 months ago
remotes/gerrit/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/gerrit/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/gerrit/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
remotes/origin/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
fatal: ambiguous argument 'ocata': unknown revision or path not in the working tree.
Use '--' to separate paths from revisions, like this:
'git <command> [<revision>...] -- [<file>...]'
remotes/origin/stable/ocata ocata-em-5-g22e9c42 ocata
remotes/origin/stable/pike 1.17.4 1.17.4 7 months ago
remotes/origin/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/origin/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/origin/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
In [5]: from imp import reload
In [6]: reload(openstack_releases.cmds.list_changes)
Out[6]: <module 'openstack_releases.cmds.list_changes' from '/home/tony/projects/openstack/openstack/releases/openstack_releases/cmds/list_changes.py'>
In [7]: openstack_releases.cmds.list_changes.git_list_existing_branches(workdir, repo)
All Branches with Version Numbers
---------------------------------
feature/add-constraints-support 1.11.0-36-g7134e3c 1.11.0 2 years, 8 months ago
master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
review/openstack_release_bot/create-stein 1.27.0-2-g31608cf 1.27.0 6 months ago
remotes/gerrit/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
remotes/gerrit/stable/ocata ocata-em-5-g22e9c42 ocata-em 2 years, 3 months ago
remotes/gerrit/stable/pike 1.17.4
remotes/gerrit/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/gerrit/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/gerrit/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
remotes/origin/master 1.27.0-10-g05a3f4d 1.27.0 6 months ago
remotes/origin/stable/ocata ocata-em-5-g22e9c42 ocata-em 2 years, 3 months ago
remotes/origin/stable/pike 1.17.4
remotes/origin/stable/queens 1.21.1-6-g02d6234 1.21.1 1 year, 1 month ago
remotes/origin/stable/rocky 1.24.1-9-g816bfc8 1.24.1 9 months ago
remotes/origin/stable/stein 1.27.0-3-g7430f4c 1.27.0 6 months ago
[1] http://logs.openstack.org/75/652775/2/check/releases-tox-list-changes/92b219a/job-output.txt.gz#_2019-04-16_14_25_50_377843
Change-Id: I6a6cf9110a737204e332c60004f72a68b770d1e7
2019-04-17 01:15:34 +00:00
|
|
|
# strip to the real tag value
|
|
|
|
match = re.match('^(.*)-[0-9]+-g[a-f0-9]+', description,
|
|
|
|
re.IGNORECASE)
|
|
|
|
if match:
|
|
|
|
tag = match.groups()[0]
|
|
|
|
else:
|
|
|
|
tag = ''
|
2017-03-20 15:17:19 +00:00
|
|
|
except subprocess.CalledProcessError as exc:
|
2018-01-09 19:53:46 +00:00
|
|
|
description = exc.output.decode('utf-8').strip()
|
|
|
|
tag = ''
|
|
|
|
if not tag:
|
|
|
|
print('{:<30} {:<20}'.format(branch, description))
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
date = subprocess.check_output(
|
|
|
|
['git', 'log', '-1', '--pretty=format:%ar', tag],
|
|
|
|
cwd=os.path.join(workdir, repo),
|
|
|
|
).decode('utf-8').strip()
|
|
|
|
except subprocess.CalledProcessError as exc:
|
|
|
|
date = exc.output.decode('utf-8')
|
|
|
|
print('{:<30} {:<20} {:<12} {}'.format(
|
|
|
|
branch, description, tag, date))
|
2017-02-16 17:10:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
def git_branch_contains(workdir, repo, title, commit):
|
|
|
|
header('%s %s' % (title, commit))
|
|
|
|
cmd = ['git', 'branch', '-r', '--contains', commit]
|
|
|
|
print('\n' + ' '.join(cmd) + '\n')
|
2017-03-03 19:46:12 +00:00
|
|
|
out = subprocess.check_output(
|
|
|
|
cmd,
|
|
|
|
cwd=os.path.join(workdir, repo),
|
|
|
|
).decode('utf-8')
|
2017-02-16 17:10:43 +00:00
|
|
|
print(out)
|
|
|
|
return sorted(
|
|
|
|
o.strip()
|
|
|
|
for o in out.splitlines()
|
|
|
|
if '->' not in o
|
|
|
|
)
|
2016-03-30 22:26:29 +00:00
|
|
|
|
|
|
|
|
2018-01-09 14:53:33 +00:00
|
|
|
def git_diff(workdir, repo, git_range, file_pattern, title=''):
|
2015-10-06 13:47:39 +00:00
|
|
|
repo_dir = os.path.join(workdir, repo)
|
|
|
|
files = list(glob.glob(os.path.join(repo_dir,
|
|
|
|
file_pattern)))
|
|
|
|
if files:
|
2018-01-09 14:53:33 +00:00
|
|
|
if title:
|
|
|
|
header(title)
|
2016-09-12 17:48:26 +00:00
|
|
|
for f in files:
|
|
|
|
cmd = [
|
|
|
|
'git', 'diff', '-U0', '--no-color',
|
|
|
|
'--ignore-space-change', '--ignore-blank-lines',
|
|
|
|
git_range,
|
|
|
|
f[len(repo_dir) + 1:],
|
|
|
|
]
|
|
|
|
print(' '.join(cmd) + '\n')
|
|
|
|
subprocess.check_call(cmd, cwd=repo_dir)
|
|
|
|
print()
|
2015-10-06 13:47:39 +00:00
|
|
|
|
|
|
|
|
2017-08-04 13:58:15 +00:00
|
|
|
def gerrit_query(*query):
|
|
|
|
url = 'https://review.openstack.org/changes/?q=' + '+'.join(query)
|
|
|
|
response = requests.get(url)
|
2017-08-04 14:20:07 +00:00
|
|
|
if (response.status_code // 100) != 2:
|
|
|
|
raise RuntimeError(
|
|
|
|
'Bad HTTP response from gerrit %s: %s' %
|
|
|
|
(url, response.status_code)
|
|
|
|
)
|
|
|
|
elif response.content[:4] == b")]}'":
|
2017-08-04 13:58:15 +00:00
|
|
|
content = response.content[5:].decode('utf-8')
|
|
|
|
return json.loads(content)
|
|
|
|
else:
|
|
|
|
print('could not parse response from %s' % url)
|
|
|
|
print(repr(content))
|
|
|
|
raise RuntimeError('failed to parse gerrit response')
|
|
|
|
|
|
|
|
|
|
|
|
def list_gerrit_patches(title, template, query):
|
|
|
|
header('{}: "{}"'.format(title, query))
|
2017-08-04 14:20:07 +00:00
|
|
|
try:
|
|
|
|
reviews = gerrit_query(query)
|
|
|
|
except Exception as err:
|
|
|
|
print(err)
|
|
|
|
else:
|
|
|
|
for r in reviews:
|
2017-08-04 14:26:34 +00:00
|
|
|
if 'topic' not in r:
|
|
|
|
r['topic'] = ''
|
2017-08-04 14:28:44 +00:00
|
|
|
try:
|
|
|
|
print(template.format(**r))
|
|
|
|
except Exception as err:
|
|
|
|
print('Could not format review data: {}'.format(err))
|
|
|
|
print(r)
|
2017-08-04 14:20:07 +00:00
|
|
|
print('{} results\n'.format(len(reviews)))
|
2017-08-04 13:58:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
def show_watched_queries(branch, repo):
|
|
|
|
with open('watched_queries.yml', 'r', encoding='utf-8') as f:
|
|
|
|
watched_queries = yamlutils.loads(f.read())
|
|
|
|
template = watched_queries['template']
|
|
|
|
for q in watched_queries['queries']:
|
|
|
|
list_gerrit_patches(
|
|
|
|
q['title'],
|
|
|
|
q.get('template', template),
|
|
|
|
q['query'].format(
|
|
|
|
branch=branch,
|
|
|
|
project=repo,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-02-08 16:53:52 +00:00
|
|
|
def show_dependency_listings(package_name, official_repos):
|
|
|
|
header('Users of {}'.format(package_name))
|
|
|
|
hound.show_dependency_listings(package_name, official_repos)
|
|
|
|
|
|
|
|
|
2015-07-23 18:17:57 +00:00
|
|
|
def main():
|
2016-09-29 16:59:56 +00:00
|
|
|
if not sys.stdout.encoding:
|
|
|
|
# Wrap sys.stdout with a writer that knows how to handle
|
|
|
|
# encoding Unicode data.
|
|
|
|
import codecs
|
|
|
|
wrapped_stdout = codecs.getwriter('UTF-8')(sys.stdout)
|
|
|
|
sys.stdout = wrapped_stdout
|
|
|
|
|
2015-07-23 18:17:57 +00:00
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument(
|
|
|
|
'--no-cleanup',
|
|
|
|
dest='cleanup',
|
|
|
|
default=True,
|
|
|
|
action='store_false',
|
|
|
|
help='do not remove temporary files',
|
|
|
|
)
|
2018-02-06 23:47:11 +00:00
|
|
|
parser.add_argument(
|
2018-04-27 21:30:34 +00:00
|
|
|
'--no-shortcut', '--force', '-f',
|
2018-02-06 23:47:11 +00:00
|
|
|
dest='shortcut',
|
|
|
|
default=True,
|
|
|
|
action='store_false',
|
|
|
|
help='if a tag has been applied, skip the repo',
|
|
|
|
)
|
2015-07-23 18:17:57 +00:00
|
|
|
parser.add_argument(
|
|
|
|
'input',
|
|
|
|
nargs='*',
|
|
|
|
help=('YAML files to validate, defaults to '
|
|
|
|
'files changed in the latest commit'),
|
|
|
|
)
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
2017-10-03 20:20:15 +00:00
|
|
|
# Set up logging, including making some loggers quiet.
|
|
|
|
logging.basicConfig(
|
2018-02-07 17:10:16 +00:00
|
|
|
format='%(levelname)7s: %(message)s',
|
2017-10-03 20:20:15 +00:00
|
|
|
stream=sys.stdout,
|
|
|
|
level=logging.DEBUG,
|
|
|
|
)
|
|
|
|
logging.getLogger('urllib3.connectionpool').setLevel(logging.WARNING)
|
|
|
|
|
2015-07-23 18:17:57 +00:00
|
|
|
filenames = args.input or gitutils.find_modified_deliverable_files()
|
|
|
|
if not filenames:
|
|
|
|
print('no modified deliverable files, skipping report')
|
|
|
|
return 0
|
|
|
|
|
|
|
|
workdir = tempfile.mkdtemp(prefix='releases-')
|
|
|
|
print('creating temporary files in %s' % workdir)
|
|
|
|
|
|
|
|
def cleanup_workdir():
|
|
|
|
if args.cleanup:
|
2018-02-11 16:51:18 +00:00
|
|
|
shutil.rmtree(workdir, True)
|
2015-07-23 18:17:57 +00:00
|
|
|
else:
|
|
|
|
print('not cleaning up %s' % workdir)
|
|
|
|
atexit.register(cleanup_workdir)
|
|
|
|
|
2018-10-12 20:05:11 +00:00
|
|
|
gov_data = governance.Governance.from_remote_repo()
|
2018-02-08 16:53:52 +00:00
|
|
|
official_repos = set(
|
|
|
|
r.name
|
2018-10-12 20:05:11 +00:00
|
|
|
for r in gov_data.get_repositories()
|
2018-02-08 16:53:52 +00:00
|
|
|
)
|
2016-07-13 16:59:47 +00:00
|
|
|
|
2018-08-17 15:37:45 +00:00
|
|
|
all_deliverables = deliverable.Deliverables(
|
|
|
|
'./deliverables',
|
|
|
|
False,
|
|
|
|
)
|
|
|
|
|
2019-11-14 23:21:32 +00:00
|
|
|
liaison_data = liaisons.get_liaisons()
|
2018-10-12 20:05:11 +00:00
|
|
|
|
2015-07-23 18:17:57 +00:00
|
|
|
# Remove any inherited PAGER environment variable to avoid
|
|
|
|
# blocking the output waiting for input.
|
|
|
|
os.environ['PAGER'] = ''
|
|
|
|
|
|
|
|
for filename in filenames:
|
2015-08-27 15:15:58 +00:00
|
|
|
if not os.path.exists(filename):
|
|
|
|
print('\n%s was removed, skipping' % filename)
|
|
|
|
continue
|
2016-02-24 17:46:46 +00:00
|
|
|
print('\n' + ('=' * 80))
|
|
|
|
print('\nChecking %s\n' % filename)
|
2018-03-08 18:16:08 +00:00
|
|
|
deliv = deliverable.Deliverable.read_file(filename)
|
2015-07-23 18:17:57 +00:00
|
|
|
|
2018-03-08 18:16:08 +00:00
|
|
|
stable_branch = deliv.series not in ['independent', defaults.RELEASE]
|
2017-11-28 16:04:26 +00:00
|
|
|
|
2016-08-26 14:44:21 +00:00
|
|
|
# By default assume the project does not use milestones.
|
2016-11-16 13:18:31 +00:00
|
|
|
header('Release model')
|
2018-03-08 18:16:08 +00:00
|
|
|
print(deliv.model)
|
2016-08-26 14:44:21 +00:00
|
|
|
|
2016-07-13 17:06:08 +00:00
|
|
|
header('Team details')
|
2018-03-08 18:16:08 +00:00
|
|
|
if deliv.team:
|
|
|
|
team_name = deliv.team
|
2018-10-12 20:05:11 +00:00
|
|
|
try:
|
|
|
|
team = gov_data.get_team(team_name)
|
|
|
|
except ValueError:
|
|
|
|
team = None
|
|
|
|
if team:
|
2016-07-13 16:59:47 +00:00
|
|
|
print('found team %s' % team_name)
|
2017-01-16 20:17:14 +00:00
|
|
|
print(' PTL : %(name)s (%(irc)s)' % team.ptl)
|
2019-06-10 16:36:56 +00:00
|
|
|
for liaison in liaison_data.get(team.name.lower(), []):
|
2019-06-20 13:56:23 +00:00
|
|
|
print(' Liaison: %(name)s (%(irc)s)' % liaison)
|
2018-03-08 18:16:08 +00:00
|
|
|
team_deliv = team.deliverables.get(deliv.name)
|
|
|
|
if team_deliv:
|
|
|
|
print('found deliverable %s' % deliv.name)
|
|
|
|
for rn, repo in sorted(team_deliv.repositories.items()):
|
2017-11-28 16:04:26 +00:00
|
|
|
follows_stable_policy = 'stable:follows-policy' in repo.tags
|
2016-07-13 16:59:47 +00:00
|
|
|
print('\nrepo %s\ntags:' % repo.name)
|
|
|
|
for t in repo.tags:
|
|
|
|
print(' %s' % t)
|
2016-07-13 17:06:08 +00:00
|
|
|
print('')
|
2017-11-28 16:04:26 +00:00
|
|
|
if stable_branch and follows_stable_policy:
|
|
|
|
banner('Needs Stable Policy Review')
|
|
|
|
print()
|
2016-07-13 16:59:47 +00:00
|
|
|
else:
|
|
|
|
print(('no deliverable %r found for team %r, '
|
|
|
|
'cannot report on governance status') %
|
2018-03-08 18:16:08 +00:00
|
|
|
(deliv.name, team_name))
|
2018-08-17 15:37:45 +00:00
|
|
|
if not deliv.is_independent:
|
|
|
|
# Show other deliverables owned by the team and
|
|
|
|
# included in this series.
|
|
|
|
team_deliv_in_series = all_deliverables.get_deliverables(
|
|
|
|
team.name, deliv.series)
|
|
|
|
if team_deliv_in_series:
|
|
|
|
print('Other {} deliverables in {}:'.format(
|
|
|
|
team.name, deliv.series))
|
|
|
|
for d in team_deliv_in_series:
|
|
|
|
print(' {} ({})'.format(d.name, d.latest_release or None))
|
|
|
|
print()
|
2016-07-13 16:59:47 +00:00
|
|
|
else:
|
|
|
|
print('no team %r found, cannot report on governance status' %
|
|
|
|
team_name)
|
|
|
|
else:
|
|
|
|
print('no team name given, cannot report on governance status')
|
|
|
|
|
2016-12-01 20:03:17 +00:00
|
|
|
# If there are no releases listed, this is probably a new
|
|
|
|
# deliverable file for initializing a new series. We don't
|
|
|
|
# need to list its changes.
|
2018-03-08 18:16:08 +00:00
|
|
|
if not deliv.is_released:
|
2016-12-01 20:03:17 +00:00
|
|
|
header('No releases')
|
|
|
|
print('no releases were found, assuming an initialization file')
|
|
|
|
continue
|
|
|
|
|
2016-08-15 20:09:44 +00:00
|
|
|
# assume the releases are in order and take the last one
|
2018-03-08 18:16:08 +00:00
|
|
|
new_release = deliv.releases[-1]
|
2016-08-15 20:09:44 +00:00
|
|
|
|
2018-03-08 18:16:08 +00:00
|
|
|
for project in new_release.projects:
|
2016-08-15 20:09:44 +00:00
|
|
|
|
2017-08-03 15:33:27 +00:00
|
|
|
tag_exists = gitutils.tag_exists(
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
|
|
|
new_release.version,
|
2016-02-03 17:13:13 +00:00
|
|
|
)
|
|
|
|
if tag_exists:
|
2016-02-24 17:46:46 +00:00
|
|
|
print('%s %s exists on git server already' %
|
2018-03-08 18:16:08 +00:00
|
|
|
(project.repo.name, new_release.version))
|
2018-02-06 23:47:11 +00:00
|
|
|
if args.shortcut:
|
|
|
|
print('skipping further processing')
|
|
|
|
continue
|
2015-07-23 18:17:57 +00:00
|
|
|
|
2018-03-08 18:16:08 +00:00
|
|
|
if project.repo.is_retired:
|
|
|
|
print('%s is retired' % (project.repo.name,))
|
2018-02-07 17:53:24 +00:00
|
|
|
if args.shortcut:
|
|
|
|
print('skipping further processing')
|
|
|
|
continue
|
|
|
|
|
2017-10-17 17:10:20 +00:00
|
|
|
# Start by checking out master, always. We need the repo
|
|
|
|
# checked out before we can tell if the stable branch
|
|
|
|
# really exists.
|
|
|
|
gitutils.clone_repo(
|
|
|
|
workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
2017-10-17 17:10:20 +00:00
|
|
|
branch='master',
|
2015-07-23 18:17:57 +00:00
|
|
|
)
|
2017-10-17 17:10:20 +00:00
|
|
|
|
2017-10-16 15:33:12 +00:00
|
|
|
# Set some git configuration values to allow us to perform
|
|
|
|
# local operations like tagging.
|
|
|
|
gitutils.ensure_basic_git_config(
|
2018-03-08 18:16:08 +00:00
|
|
|
workdir, project.repo.name,
|
2017-10-16 15:33:12 +00:00
|
|
|
{'user.email': 'openstack-infra@lists.openstack.org',
|
|
|
|
'user.name': 'OpenStack Proposal Bot'},
|
|
|
|
)
|
2015-08-13 18:22:34 +00:00
|
|
|
|
2017-08-22 19:48:17 +00:00
|
|
|
# Determine which branch we should actually be looking
|
|
|
|
# at. Assume any series for which there is no stable
|
|
|
|
# branch will be on 'master'.
|
2018-03-08 18:16:08 +00:00
|
|
|
if gitutils.stable_branch_exists(workdir,
|
|
|
|
project.repo.name,
|
|
|
|
deliv.series):
|
|
|
|
branch = 'stable/' + deliv.series
|
2017-08-22 19:48:17 +00:00
|
|
|
else:
|
|
|
|
branch = 'master'
|
|
|
|
|
2017-10-17 17:10:20 +00:00
|
|
|
if branch != 'master':
|
2017-08-22 19:48:17 +00:00
|
|
|
# Check out the repo again to the right branch if we
|
|
|
|
# didn't get it the first time.
|
2017-10-17 17:10:20 +00:00
|
|
|
gitutils.clone_repo(
|
|
|
|
workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
2017-10-17 17:10:20 +00:00
|
|
|
branch=branch,
|
2017-08-22 19:48:17 +00:00
|
|
|
)
|
|
|
|
|
2016-08-15 20:09:44 +00:00
|
|
|
# look at the previous tag for the parent of the commit
|
|
|
|
# getting the new release
|
|
|
|
previous_tag = gitutils.get_latest_tag(
|
|
|
|
workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
2019-01-07 15:26:36 +00:00
|
|
|
'{}^'.format(project.hash),
|
|
|
|
always=False,
|
2016-08-15 20:09:44 +00:00
|
|
|
)
|
2018-03-08 18:16:08 +00:00
|
|
|
try:
|
|
|
|
previous_release = deliv.get_release(previous_tag)
|
|
|
|
except ValueError:
|
|
|
|
previous_release = None
|
2016-08-15 20:09:44 +00:00
|
|
|
|
|
|
|
start_range = previous_tag
|
2015-11-25 18:01:32 +00:00
|
|
|
if previous_release:
|
2015-11-25 20:27:36 +00:00
|
|
|
previous_project = {
|
2018-03-08 18:16:08 +00:00
|
|
|
x.repo.name: x
|
|
|
|
for x in previous_release.projects
|
|
|
|
}.get(project.repo.name)
|
2015-11-25 20:27:36 +00:00
|
|
|
if previous_project is not None:
|
2016-08-15 20:09:44 +00:00
|
|
|
start_range = previous_tag
|
2015-08-24 18:17:01 +00:00
|
|
|
|
|
|
|
if start_range:
|
2018-03-08 18:16:08 +00:00
|
|
|
git_range = '%s..%s' % (start_range, project.hash)
|
2015-08-24 18:17:01 +00:00
|
|
|
else:
|
2018-03-08 18:16:08 +00:00
|
|
|
git_range = project.hash
|
2015-08-24 18:17:01 +00:00
|
|
|
|
2016-02-29 15:15:57 +00:00
|
|
|
# Show details about the commit being tagged.
|
2016-07-13 17:00:56 +00:00
|
|
|
header('Details for commit receiving new tag %s' %
|
2018-03-08 18:16:08 +00:00
|
|
|
new_release.version)
|
|
|
|
print('\ngit describe %s\n' % project.hash)
|
2016-02-29 15:15:57 +00:00
|
|
|
try:
|
|
|
|
subprocess.check_call(
|
2018-03-08 18:16:08 +00:00
|
|
|
['git', 'describe', project.hash],
|
|
|
|
cwd=os.path.join(workdir, project.repo.name),
|
2016-02-29 15:15:57 +00:00
|
|
|
)
|
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
print('WARNING: Could not run git describe: %s' % e)
|
|
|
|
|
2020-09-03 12:34:49 +00:00
|
|
|
header('reno SemVer suggestion')
|
|
|
|
print('\n(This may produce an old tag version if no release notes were added.\n')
|
|
|
|
print('\nreno -q semver-next\n')
|
|
|
|
try:
|
|
|
|
subprocess.check_call(
|
|
|
|
['reno', '-q', 'semver-next'],
|
|
|
|
cwd=os.path.join(workdir, project.repo.name),
|
|
|
|
)
|
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
print('WARNING: Could not run reno semver-next: %s' % e)
|
|
|
|
|
2016-02-29 15:15:57 +00:00
|
|
|
git_show(
|
|
|
|
workdir=workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
repo=project.repo.name,
|
2016-02-29 15:15:57 +00:00
|
|
|
title='Check existing tags',
|
2018-03-08 18:16:08 +00:00
|
|
|
ref=project.hash,
|
2016-02-29 15:15:57 +00:00
|
|
|
)
|
|
|
|
|
2017-02-16 17:10:43 +00:00
|
|
|
git_list_existing_branches(
|
|
|
|
workdir=workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
repo=project.repo.name,
|
2017-02-16 17:10:43 +00:00
|
|
|
)
|
|
|
|
|
2016-08-15 20:09:44 +00:00
|
|
|
branches = git_branch_contains(
|
2016-03-30 22:26:29 +00:00
|
|
|
workdir=workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
repo=project.repo.name,
|
2016-03-30 22:26:29 +00:00
|
|
|
title='Branches containing commit',
|
2018-03-08 18:16:08 +00:00
|
|
|
commit=project.hash,
|
2016-03-30 22:26:29 +00:00
|
|
|
)
|
|
|
|
|
2016-08-15 20:09:44 +00:00
|
|
|
header('Relationship to HEAD')
|
2018-03-08 18:16:08 +00:00
|
|
|
if deliv.is_independent:
|
2017-06-05 18:54:05 +00:00
|
|
|
if branches:
|
|
|
|
tag_branch = branches[0]
|
|
|
|
else:
|
|
|
|
tag_branch = branch
|
2016-08-15 20:09:44 +00:00
|
|
|
head_sha = gitutils.sha_for_tag(
|
|
|
|
workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
2016-09-13 18:47:05 +00:00
|
|
|
tag_branch,
|
2016-08-15 20:09:44 +00:00
|
|
|
)
|
2016-09-13 18:47:05 +00:00
|
|
|
print('HEAD of {} is {}'.format(tag_branch, head_sha))
|
2016-08-15 20:09:44 +00:00
|
|
|
else:
|
2017-03-13 18:46:49 +00:00
|
|
|
if (branch in branches) or (not branches):
|
2017-02-16 17:12:11 +00:00
|
|
|
tag_branch = branch
|
|
|
|
else:
|
|
|
|
tag_branch = branches[0]
|
2016-08-15 20:09:44 +00:00
|
|
|
head_sha = gitutils.sha_for_tag(
|
|
|
|
workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
2017-02-16 17:12:11 +00:00
|
|
|
tag_branch,
|
2016-08-15 20:09:44 +00:00
|
|
|
)
|
2017-02-16 17:12:11 +00:00
|
|
|
print('HEAD of {} is {}'.format(tag_branch, head_sha))
|
2016-03-03 21:32:58 +00:00
|
|
|
requested_sha = gitutils.sha_for_tag(
|
|
|
|
workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
|
|
|
project.hash,
|
2016-03-03 21:32:58 +00:00
|
|
|
)
|
|
|
|
# If the sha for HEAD and the requested release don't
|
|
|
|
# match, show any unreleased changes on the branch. We ask
|
|
|
|
# git to give us the real SHA for the requested release in
|
|
|
|
# case the deliverables file has the short version of the
|
|
|
|
# hash.
|
|
|
|
if head_sha == requested_sha:
|
2016-09-13 18:47:05 +00:00
|
|
|
print('\nRequest releases from HEAD on %s' % tag_branch)
|
2016-03-03 21:32:58 +00:00
|
|
|
else:
|
2018-03-08 18:16:08 +00:00
|
|
|
git_log(workdir, project.repo.name, 'Release will NOT include',
|
2016-08-15 20:09:44 +00:00
|
|
|
'%s..%s' % (requested_sha, head_sha),
|
2016-03-03 21:32:58 +00:00
|
|
|
extra_args=['--format=%h %ci %s'])
|
|
|
|
|
2018-03-08 18:16:08 +00:00
|
|
|
show_watched_queries(branch, project.repo.name)
|
2017-08-04 13:58:15 +00:00
|
|
|
|
2015-10-06 13:47:39 +00:00
|
|
|
# Show any requirements changes in the upcoming release.
|
2016-10-07 19:47:00 +00:00
|
|
|
# Include setup.cfg, in case the project uses "extras".
|
2016-01-21 18:34:39 +00:00
|
|
|
if start_range:
|
2018-03-08 18:16:08 +00:00
|
|
|
git_diff(workdir, project.repo.name, git_range, '*requirements*.txt',
|
2018-01-09 14:53:33 +00:00
|
|
|
'Requirements Changes %s' % git_range)
|
2018-03-08 18:16:08 +00:00
|
|
|
git_diff(workdir, project.repo.name, git_range, 'doc/requirements.txt',
|
2018-01-09 14:58:32 +00:00
|
|
|
'Doc Requirements Changes %s' % git_range)
|
2018-03-08 18:16:08 +00:00
|
|
|
git_diff(workdir, project.repo.name, git_range, 'setup.cfg',
|
2018-01-09 14:53:33 +00:00
|
|
|
'setup.cfg Changes %s' % git_range)
|
2018-03-08 18:16:08 +00:00
|
|
|
git_diff(workdir, project.repo.name, git_range, 'bindep.txt',
|
2018-01-09 14:54:24 +00:00
|
|
|
'bindep.txt Changes %s' % git_range)
|
2015-10-06 13:47:39 +00:00
|
|
|
|
2017-10-16 13:44:46 +00:00
|
|
|
# Before we try to determine if the previous release
|
|
|
|
# is an ancestor or produce the release notes we need
|
|
|
|
# the tag to exist in the local repository.
|
|
|
|
if not tag_exists:
|
|
|
|
header('Applying Temporary Tag')
|
|
|
|
print('\ngit tag {version} {hash}'.format(
|
2018-03-08 18:16:08 +00:00
|
|
|
version=new_release.version,
|
|
|
|
hash=project.hash,
|
2017-10-16 13:44:46 +00:00
|
|
|
))
|
|
|
|
subprocess.check_call(
|
2018-03-08 18:16:08 +00:00
|
|
|
['git', 'tag', new_release.version,
|
|
|
|
project.hash],
|
|
|
|
cwd=os.path.join(workdir, project.repo.name),
|
2017-10-16 13:44:46 +00:00
|
|
|
)
|
|
|
|
|
2016-02-03 17:13:13 +00:00
|
|
|
# Show any changes in the previous release but not in this
|
|
|
|
# release, in case someone picks an "early" SHA or a
|
|
|
|
# regular commit instead of the appropriate merge commit.
|
2016-02-29 21:37:03 +00:00
|
|
|
previous_tag_exists = False
|
2016-02-03 17:13:13 +00:00
|
|
|
if previous_release:
|
2017-08-03 15:33:27 +00:00
|
|
|
previous_tag_exists = gitutils.tag_exists(
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
|
|
|
previous_release.version,
|
2016-02-29 21:37:03 +00:00
|
|
|
)
|
|
|
|
if previous_tag_exists:
|
2016-02-03 17:13:13 +00:00
|
|
|
git_log(
|
2018-03-08 18:16:08 +00:00
|
|
|
workdir, project.repo.name,
|
2016-02-03 17:13:13 +00:00
|
|
|
'Patches in previous release but not in this one',
|
2018-03-08 18:16:08 +00:00
|
|
|
[previous_release.version,
|
2016-02-03 17:13:13 +00:00
|
|
|
'--not',
|
2018-03-08 18:16:08 +00:00
|
|
|
project.hash],
|
2016-02-03 17:13:13 +00:00
|
|
|
extra_args=['--topo-order', '--oneline', '--no-merges'],
|
|
|
|
)
|
2017-10-16 13:44:46 +00:00
|
|
|
|
|
|
|
# The tag will have been added as a local tag above if
|
|
|
|
# it does not already exist.
|
2016-02-03 17:13:13 +00:00
|
|
|
header('New release %s includes previous release %s' %
|
2018-03-08 18:16:08 +00:00
|
|
|
(new_release.version, previous_release.version))
|
2016-02-03 17:13:13 +00:00
|
|
|
print('\ngit tag --contains %s\n' %
|
2018-03-08 18:16:08 +00:00
|
|
|
previous_release.version)
|
2016-02-03 17:13:13 +00:00
|
|
|
containing_tags = subprocess.check_output(
|
|
|
|
['git', 'tag',
|
|
|
|
'--contains',
|
2018-03-08 18:16:08 +00:00
|
|
|
previous_release.version],
|
|
|
|
cwd=os.path.join(workdir, project.repo.name),
|
2017-03-03 19:46:12 +00:00
|
|
|
).decode('utf-8').split()
|
2016-02-03 17:13:13 +00:00
|
|
|
print('Containing tags:', containing_tags)
|
2018-03-08 18:16:08 +00:00
|
|
|
if new_release.version not in containing_tags:
|
|
|
|
print('WARNING: Missing %s' % new_release.version)
|
2016-02-03 17:13:13 +00:00
|
|
|
else:
|
2018-03-08 18:16:08 +00:00
|
|
|
print('Found new version %s' % new_release.version)
|
2016-02-03 17:13:13 +00:00
|
|
|
|
|
|
|
is_ancestor = gitutils.check_ancestry(
|
|
|
|
workdir,
|
2018-03-08 18:16:08 +00:00
|
|
|
project.repo.name,
|
|
|
|
previous_release.version,
|
|
|
|
project.hash,
|
2016-02-03 17:13:13 +00:00
|
|
|
)
|
|
|
|
if is_ancestor:
|
|
|
|
print('SHA found in descendants')
|
|
|
|
else:
|
|
|
|
print('SHA NOT FOUND in descendants')
|
|
|
|
|
2017-08-22 20:27:15 +00:00
|
|
|
# Show the changes since the last release, first as a
|
|
|
|
# graph view so we can check for bad merges, and then with
|
|
|
|
# more detail.
|
2018-03-08 18:16:08 +00:00
|
|
|
git_log(workdir, project.repo.name,
|
|
|
|
'Release %s will include' % new_release.version,
|
2017-08-22 20:27:15 +00:00
|
|
|
git_range,
|
|
|
|
extra_args=['--graph', '--oneline', '--decorate',
|
|
|
|
'--topo-order'])
|
2018-03-08 18:16:08 +00:00
|
|
|
git_log(workdir, project.repo.name,
|
2017-08-22 20:27:15 +00:00
|
|
|
'Details Contents',
|
|
|
|
git_range,
|
|
|
|
extra_args=['--no-merges', '--topo-order'])
|
|
|
|
|
2017-10-16 13:44:46 +00:00
|
|
|
# The tag will have been added as a local tag above if it does
|
|
|
|
# not already exist.
|
2017-10-03 20:21:30 +00:00
|
|
|
header('Release Notes')
|
|
|
|
try:
|
|
|
|
notes = release_notes.generate_release_notes(
|
2018-03-08 18:16:08 +00:00
|
|
|
repo=project.repo.name,
|
|
|
|
repo_path=os.path.join(workdir, project.repo.name),
|
2019-01-07 15:26:36 +00:00
|
|
|
start_revision=new_release.diff_start or start_range or '',
|
2018-03-08 18:16:08 +00:00
|
|
|
end_revision=new_release.version,
|
2017-10-03 20:21:30 +00:00
|
|
|
show_dates=True,
|
|
|
|
skip_requirement_merges=True,
|
|
|
|
is_stable=branch.startswith('stable/'),
|
2018-03-08 18:16:08 +00:00
|
|
|
series=deliv.series,
|
2017-10-03 20:21:30 +00:00
|
|
|
email='test-job@openstack.org',
|
|
|
|
email_from='test-job@openstack.org',
|
|
|
|
email_reply_to='noreply@openstack.org',
|
|
|
|
email_tags='',
|
|
|
|
include_pypi_link=False,
|
|
|
|
changes_only=False,
|
2018-03-08 18:16:08 +00:00
|
|
|
first_release=deliv.is_first_release,
|
2019-04-16 16:05:51 +00:00
|
|
|
deliverable_file=filename,
|
2017-10-03 20:21:30 +00:00
|
|
|
description='',
|
2018-03-08 18:16:08 +00:00
|
|
|
publishing_dir_name=project.repo.name,
|
2017-10-03 20:21:30 +00:00
|
|
|
)
|
2020-07-27 21:20:03 +00:00
|
|
|
except Exception:
|
2017-10-03 20:21:30 +00:00
|
|
|
logging.exception('Failed to produce release notes')
|
|
|
|
else:
|
|
|
|
print('\n')
|
|
|
|
print(notes)
|
|
|
|
|
2018-03-08 18:16:08 +00:00
|
|
|
if 'library' in deliv.type:
|
2018-02-08 16:53:52 +00:00
|
|
|
show_dependency_listings(
|
2018-03-08 18:16:08 +00:00
|
|
|
project.guess_sdist_name(),
|
2018-02-08 16:53:52 +00:00
|
|
|
official_repos,
|
|
|
|
)
|
|
|
|
|
2015-07-23 18:17:57 +00:00
|
|
|
return 0
|