releases/tools/delete_unmaintained_branch.py
Dr. Jens Harbott 54c6dd50da Update tooling to delete unmaintained branches
The tools/list_eol_stale_branches.sh script needs to handle
unmaintained/* branches in addition to stable/* branches, the former
also need to be deleted once the matching *-eol tag has been added to a
repository. In addition it needs to handle the delta between series and
branch names starting with 2023.1/antelope.

Also add a script to abandon open reviews on unmaintained branches, which
is a requirement in order to be able to delete them.

Change-Id: Iba0a50ad0d92013362e2a5eca5a4b1968e6acaf9
2024-12-11 14:40:00 +01:00

115 lines
4.4 KiB
Python
Executable File

#!/usr/bin/env python3
# Copyright 2021 Ericsson Software Technology
#
# 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.
# This script helps deleting unmaintained branches. Requires 'Delete reference'
# access category for 'refs/for/unmaintained/*' for the user who executes it.
import argparse
import getpass
import json
import sys
from oslo_utils import timeutils
import requests
GERRIT_URL = 'https://review.opendev.org/'
GITEA_URL = 'https://opendev.org/api/v1'
def delete_branch(username, project_name, branch_id):
print(f'!!! WARNING: You are about to delete unmaintained/{branch_id} '
f'from {project_name} !!!')
gerrit_auth = requests.auth.HTTPBasicAuth(
username,
getpass.getpass('Gerrit password: '))
url = (f'{GERRIT_URL}a/projects/{project_name.replace("/", "%2F")}/'
f'branches/unmaintained%2F{branch_id}')
response = requests.delete(url, auth=gerrit_auth)
if response.status_code == 204:
print(f'{timeutils.utcnow()} | Branch unmaintained/{branch_id} successfully deleted '
f'from {project_name}!')
return 0
elif response.status_code == 401:
print(f'{timeutils.utcnow()} | 401 Unauthorized.')
return 1
else:
# NOTE(elod.illes): other possible errors from gerrit:
# 404: In case of project or branch is not found
# 409: Branch has open changes
print(f'{timeutils.utcnow()} | Delete failed ({response.status_code}): {response.text}')
return 2
def is_branch_open(project_name, branch_id, quiet):
url = (f'{GITEA_URL}/repos/{project_name.replace("/", "%2F")}/'
f'branches/unmaintained%2F{branch_id}')
response = requests.get(url)
try:
response_details = response.json()
except json.decoder.JSONDecodeError as exc:
print(f'ERROR: JSON decode failed ({exc})')
print(f'ERROR: ({response.status_code}): {response.text}')
print('Is the project name correct, like "openstack/nova"?')
return 4
if response.status_code == 200:
if not quiet:
print(f'unmaintained/{branch_id} exists in {project_name}.')
return 0
elif ((response.status_code == 404) and
(response_details['errors'] == [f'branch does not exist [name: unmaintained/{branch_id}]']) and
(response_details['message'] == "The target couldn't be found.")):
if not quiet:
print(f'unmaintained/{branch_id} does not exist in {project_name}.')
return 1
else:
print(f'ERROR: ({response.status_code}): {response.text}')
return 2
def main():
parser = argparse.ArgumentParser(
description='Deletes unmaintained/<branch> from <project>.')
subparsers = parser.add_subparsers(required=True, dest='command',
metavar='{delete,check}')
delete_parser = subparsers.add_parser(
'delete', help='Delete unmaintained/<branch> from <project>')
delete_parser.add_argument('username', help='Gerrit Username')
delete_parser.add_argument('project', help='Project to delete from')
delete_parser.add_argument('branch', help='Branch to delete')
check_parser = subparsers.add_parser(
'check', help='Check if unmaintained/<branch> exists for <project>')
check_parser.add_argument('project', help='Project to check')
check_parser.add_argument('branch', help='Branch to check if exists')
check_parser.add_argument(
'-q', '--quiet', action='store_true',
help='Return code only (0 means branch exists)')
args = parser.parse_args()
if args.command == 'delete':
return delete_branch(args.username,
args.project,
args.branch.replace('unmaintained/', ''))
elif args.command == 'check':
return is_branch_open(args.project,
args.branch.replace('unmaintained/', ''),
args.quiet)
if __name__ == '__main__':
sys.exit(main())