5954003e8d
This reverts commit df387cafb0
.
Reason for revert: Nothing in centos provides cliff. This cannot be satisfied during the initial installation. You cannot depend on cliff.
Change-Id: Ib79bd0f1a00ee95bfdae2c1ed4976d0c674c5280
545 lines
20 KiB
Python
Executable File
545 lines
20 KiB
Python
Executable File
#!/usr/bin/env python
|
|
|
|
# Copyright 2016 Red Hat, Inc.
|
|
# 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.
|
|
|
|
from __future__ import print_function
|
|
import argparse
|
|
import os
|
|
import re
|
|
import subprocess
|
|
import sys
|
|
|
|
import requests
|
|
|
|
|
|
TITLE_RE = re.compile('\\[(.*)\\]')
|
|
NAME_RE = re.compile('name=(.+)')
|
|
PRIORITY_RE = re.compile('priority=\\d+')
|
|
# Packages to be included from delorean-current when using current-tripleo
|
|
INCLUDE_PKGS = ('includepkgs=instack,instack-undercloud,'
|
|
'os-apply-config,os-collect-config,os-net-config,'
|
|
'os-refresh-config,python*-tripleoclient,'
|
|
'openstack-tripleo-*,openstack-puppet-modules,'
|
|
'ansible-role-tripleo*,puppet-*,python*-tripleo-common,'
|
|
'python*-paunch*,tripleo-ansible,ansible-config_template')
|
|
DEFAULT_OUTPUT_PATH = '/etc/yum.repos.d'
|
|
DEFAULT_RDO_MIRROR = 'https://trunk.rdoproject.org'
|
|
|
|
# RHEL is only provided to licensed cloud providers via RHUI
|
|
DEFAULT_MIRROR_MAP = {
|
|
'fedora': 'https://mirrors.fedoraproject.org',
|
|
'centos': 'http://mirror.centos.org',
|
|
'ubi': 'http://mirror.centos.org',
|
|
'rhel': 'https://trunk.rdoproject.org',
|
|
}
|
|
CEPH_REPO_TEMPLATE = '''
|
|
[tripleo-centos-ceph-%(ceph_release)s]
|
|
name=tripleo-centos-ceph-%(ceph_release)s
|
|
baseurl=%(mirror)s/centos/%(centos_release)s/storage/$basearch/ceph-%(ceph_release)s/
|
|
gpgcheck=0
|
|
enabled=1
|
|
'''
|
|
OPSTOOLS_REPO_TEMPLATE = '''
|
|
[tripleo-centos-opstools]
|
|
name=tripleo-centos-opstools
|
|
baseurl=%(mirror)s/centos/7/opstools/$basearch/
|
|
gpgcheck=0
|
|
enabled=1
|
|
'''
|
|
# centos-8 only
|
|
HIGHAVAILABILITY_REPO_TEMPLATE = '''
|
|
[tripleo-centos-highavailability]
|
|
name=tripleo-centos-highavailability
|
|
baseurl=%(mirror)s/centos/%(stream)s/HighAvailability/$basearch/os/
|
|
gpgcheck=0
|
|
enabled=1
|
|
'''
|
|
# centos-8 only
|
|
POWERTOOLS_REPO_TEMPLATE = '''
|
|
[tripleo-centos-powertools]
|
|
name=tripleo-centos-powertools
|
|
baseurl=%(mirror)s/centos/%(stream)s/PowerTools/$basearch/os/
|
|
gpgcheck=0
|
|
enabled=1
|
|
'''
|
|
# ubi-8 only
|
|
APPSTREAM_REPO_TEMPLATE = '''
|
|
[AppStream]
|
|
name=CentOS-$releasever - AppStream
|
|
baseurl=%(mirror)s/centos/$releasever/AppStream/$basearch/os/
|
|
gpgcheck=0
|
|
enabled=1
|
|
%(extra)s
|
|
'''
|
|
BASE_REPO_TEMPLATE = '''
|
|
[BaseOS]
|
|
name=CentOS-$releasever - Base
|
|
baseurl=%(mirror)s/centos/$releasever/BaseOS/$basearch/os/
|
|
gpgcheck=0
|
|
enabled=1
|
|
'''
|
|
|
|
|
|
# unversioned fedora added for backwards compatibility
|
|
SUPPORTED_DISTROS = [
|
|
('centos', '7'),
|
|
('centos', '8'),
|
|
('fedora', ''),
|
|
('rhel', '8'),
|
|
('ubi', '8') # a subcase of the rhel distro
|
|
]
|
|
|
|
|
|
class InvalidArguments(Exception):
|
|
pass
|
|
|
|
|
|
class NoRepoTitle(Exception):
|
|
pass
|
|
|
|
|
|
def _get_distro():
|
|
"""Get distro info from os-release
|
|
|
|
returns: distro_id, distro_major_version_id, distro_name
|
|
"""
|
|
output = subprocess.Popen(
|
|
'source /etc/os-release && echo -e -n "$ID\n$VERSION_ID\n$NAME"',
|
|
shell=True,
|
|
stdout=subprocess.PIPE,
|
|
stderr=open(os.devnull, 'w'),
|
|
executable='/bin/bash',
|
|
universal_newlines=True).communicate()
|
|
|
|
# distro_id and distro_version_id will always be at least an empty string
|
|
distro_id, distro_version_id, distro_name = output[0].split('\n')
|
|
|
|
# if distro_version_id is empty string the major version will be empty
|
|
# string too
|
|
distro_major_version_id = distro_version_id.split('.')[0]
|
|
|
|
# check if that is UBI subcase?
|
|
if os.path.exists('/etc/yum.repos.d/ubi.repo'):
|
|
distro_id = 'ubi'
|
|
|
|
if (distro_id, distro_major_version_id) not in SUPPORTED_DISTROS:
|
|
print(
|
|
"WARNING: Unsupported platform '{}{}' detected by tripleo-repos,"
|
|
" centos7 will be used unless you use CLI param to change it."
|
|
"".format(distro_id, distro_major_version_id), file=sys.stderr)
|
|
distro_id = 'centos'
|
|
distro_major_version_id = '7'
|
|
|
|
if distro_id == 'ubi':
|
|
print(
|
|
"WARNING: Centos{} Base and AppStream will be installed for "
|
|
"this UBI distro".format(distro_major_version_id))
|
|
|
|
return distro_id, distro_major_version_id, distro_name
|
|
|
|
|
|
def _parse_args(distro_id, distro_major_version_id):
|
|
|
|
distro = "{}{}".format(distro_id, distro_major_version_id)
|
|
|
|
# Calculating arguments default from constants
|
|
default_mirror = DEFAULT_MIRROR_MAP[distro_id]
|
|
distro_choices = ["".join(distro_pair)
|
|
for distro_pair in SUPPORTED_DISTROS]
|
|
|
|
parser = argparse.ArgumentParser(
|
|
description='Download and install repos necessary for TripleO. Note '
|
|
'that some of these repos require yum-plugin-priorities, '
|
|
'so that will also be installed.',
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
|
|
parser.add_argument('repos', metavar='REPO', nargs='+',
|
|
choices=['current', 'deps', 'current-tripleo',
|
|
'current-tripleo-dev', 'ceph', 'opstools',
|
|
'tripleo-ci-testing', 'current-tripleo-rdo'],
|
|
help='A list of repos. Available repos: '
|
|
'%(choices)s. The deps repo will always be '
|
|
'included when using current or '
|
|
'current-tripleo. current-tripleo-dev '
|
|
'downloads the current-tripleo, current, and '
|
|
'deps repos, but sets the current repo to only '
|
|
'be used for TripleO projects. It also modifies '
|
|
'each repo\'s priority so packages are installed '
|
|
'from the appropriate location.')
|
|
parser.add_argument('-d', '--distro',
|
|
default=distro,
|
|
choices=distro_choices,
|
|
nargs='?',
|
|
help='Target distro with default detected at runtime. '
|
|
)
|
|
parser.add_argument('-b', '--branch',
|
|
default='master',
|
|
help='Target branch. Should be the lowercase name of '
|
|
'the OpenStack release. e.g. liberty')
|
|
parser.add_argument('-o', '--output-path',
|
|
default=DEFAULT_OUTPUT_PATH,
|
|
help='Directory in which to save the selected repos.')
|
|
parser.add_argument('--mirror',
|
|
default=default_mirror,
|
|
help='Server from which to install base OS packages. '
|
|
'Default value is based on distro param.')
|
|
parser.add_argument('--rdo-mirror',
|
|
default=DEFAULT_RDO_MIRROR,
|
|
help='Server from which to install RDO packages.')
|
|
stream_group = parser.add_mutually_exclusive_group()
|
|
stream_group.add_argument('--stream',
|
|
action='store_true',
|
|
default=True,
|
|
help='Enable stream support for CentOS repos')
|
|
stream_group.add_argument('--no-stream',
|
|
action='store_true',
|
|
default=False,
|
|
help='Disable stream support for CentOS repos')
|
|
|
|
args = parser.parse_args()
|
|
if args.no_stream:
|
|
args.stream = False
|
|
args.old_mirror = default_mirror
|
|
|
|
return args
|
|
|
|
|
|
def _get_repo(path, args):
|
|
r = requests.get(path)
|
|
if r.status_code == 200:
|
|
return _inject_mirrors(r.text, args)
|
|
else:
|
|
r.raise_for_status()
|
|
|
|
|
|
def _write_repo(content, target, name=None):
|
|
if not name:
|
|
m = TITLE_RE.search(content)
|
|
if not m:
|
|
raise NoRepoTitle('Could not find repo title in: \n%s' % content)
|
|
name = m.group(1)
|
|
# centos-8 dlrn repos have changed. repos per component
|
|
# are folded into a single repo.
|
|
if 'component' in name:
|
|
name = 'delorean'
|
|
filename = name + '.repo'
|
|
filename = os.path.join(target, filename)
|
|
with open(filename, 'w') as f:
|
|
f.write(content)
|
|
print('Installed repo %s to %s' % (name, filename))
|
|
|
|
|
|
def _validate_distro_repos(args):
|
|
"""Validate requested repos are valid for the distro"""
|
|
valid_repos = []
|
|
if 'fedora' in args.distro:
|
|
valid_repos = ['current', 'current-tripleo', 'ceph', 'deps',
|
|
'tripleo-ci-testing']
|
|
elif args.distro in ['centos7', 'centos8', 'rhel8', 'ubi8']:
|
|
valid_repos = ['ceph', 'current', 'current-tripleo',
|
|
'current-tripleo-dev', 'deps', 'tripleo-ci-testing',
|
|
'opstools', 'current-tripleo-rdo']
|
|
invalid_repos = [x for x in args.repos if x not in valid_repos]
|
|
if len(invalid_repos) > 0:
|
|
raise InvalidArguments('{} repo(s) are not valid for {}. Valid repos '
|
|
'are: {}'.format(invalid_repos, args.distro,
|
|
valid_repos))
|
|
return True
|
|
|
|
|
|
def _validate_current_tripleo(repos):
|
|
"""Validate current usage
|
|
|
|
current and current-tripleo cannot be specified with each other and
|
|
current-tripleo-dev is a mix of current, current-tripleo and deps
|
|
so they should not be specified on the command line with each other.
|
|
"""
|
|
if 'current-tripleo' in repos and 'current' in repos:
|
|
raise InvalidArguments('Cannot use current and current-tripleo at the '
|
|
'same time.')
|
|
if 'current-tripleo-dev' not in repos:
|
|
return True
|
|
if 'current' in repos or 'current-tripleo' in repos or 'deps' in repos:
|
|
raise InvalidArguments('current-tripleo-dev should not be used with '
|
|
'any other RDO Trunk repos.')
|
|
return True
|
|
|
|
|
|
def _validate_tripleo_ci_testing(repos):
|
|
"""Validate tripleo-ci-testing
|
|
|
|
With tripleo-ci-testing for repo (currently only periodic container build)
|
|
no other repos expected except optionally deps|ceph|opstools
|
|
which is enabled regardless.
|
|
"""
|
|
if 'tripleo-ci-testing' in repos and len(repos) > 1:
|
|
if 'deps' in repos or 'ceph' in repos or 'opstools' in repos:
|
|
return True
|
|
else:
|
|
raise InvalidArguments('Cannot use tripleo-ci-testing at the '
|
|
'same time as other repos, except '
|
|
'deps|ceph|opstools.')
|
|
return True
|
|
|
|
|
|
def _validate_distro_stream(args, distro_name):
|
|
"""Validate stream related args vs host
|
|
|
|
Fails if stream is to be used but the host isn't a stream OS or vice versa
|
|
"""
|
|
is_stream = args.stream and not args.no_stream
|
|
if is_stream and 'stream' not in distro_name.lower():
|
|
raise InvalidArguments('--stream provided, but OS is not the Stream '
|
|
'version. Please ensure the host is Stream.')
|
|
elif not is_stream and 'stream' in distro_name.lower():
|
|
raise InvalidArguments('--no-stream provided, but OS is the Stream '
|
|
'version. Please ensure the host is not the '
|
|
'Stream version.')
|
|
return True
|
|
|
|
|
|
def _validate_args(args, distro_name):
|
|
_validate_current_tripleo(args.repos)
|
|
_validate_distro_repos(args)
|
|
_validate_tripleo_ci_testing(args.repos)
|
|
_validate_distro_stream(args, distro_name)
|
|
|
|
|
|
def _remove_existing(args):
|
|
"""Remove any delorean* or opstools repos that already exist"""
|
|
if args.distro == 'ubi8':
|
|
regex = '^(BaseOS|AppStream|delorean|tripleo-centos-' \
|
|
'(opstools|ceph|highavailability|powertools)).*.repo'
|
|
else:
|
|
regex = '^(delorean|tripleo-centos-' \
|
|
'(opstools|ceph|highavailability|powertools)).*.repo'
|
|
pattern = re.compile(regex)
|
|
if os.path.exists("/etc/distro.repos.d"):
|
|
paths = set(
|
|
os.listdir(args.output_path) + os.listdir("/etc/distro.repos.d"))
|
|
else:
|
|
paths = os.listdir(args.output_path)
|
|
for f in paths:
|
|
if pattern.match(f):
|
|
filename = os.path.join(args.output_path, f)
|
|
if os.path.exists(filename):
|
|
os.remove(filename)
|
|
print('Removed old repo "%s"' % filename)
|
|
filename = os.path.join("/etc/distro.repos.d", f)
|
|
if os.path.exists(filename):
|
|
os.remove(filename)
|
|
print('Removed old repo "%s"' % filename)
|
|
|
|
|
|
def _get_base_path(args):
|
|
if args.distro == 'ubi8':
|
|
distro = 'centos8' # there are no base paths for UBI that work well
|
|
else:
|
|
distro = args.distro
|
|
|
|
# The mirror url with /$DISTRO$VERSION path for master branch is
|
|
# deprecated.
|
|
# The default for rdo mirrors is $DISTRO$VERSION-$BRANCH
|
|
# it should work for every (distro, branch) pair that
|
|
# makes sense
|
|
# Any exception should be corrected at source, not here.
|
|
distro_branch = '%s-%s' % (distro, args.branch)
|
|
return '%s/%s/' % (args.rdo_mirror, distro_branch)
|
|
|
|
|
|
def _install_priorities():
|
|
try:
|
|
subprocess.check_call(['yum', 'install', '-y',
|
|
'yum-plugin-priorities'])
|
|
except subprocess.CalledProcessError as e:
|
|
print('ERROR: Failed to install yum-plugin-priorities\n%s\n%s' %
|
|
(e.cmd, e.output))
|
|
raise
|
|
|
|
|
|
def _create_ceph(args, release):
|
|
"""Generate a Ceph repo file for release"""
|
|
centos_release = '7' if args.distro == 'centos7' else '8'
|
|
return CEPH_REPO_TEMPLATE % {'centos_release': centos_release,
|
|
'ceph_release': release,
|
|
'mirror': args.mirror}
|
|
|
|
|
|
def _change_priority(content, new_priority):
|
|
new_content = PRIORITY_RE.sub('priority=%d' % new_priority, content)
|
|
# This shouldn't happen, but let's be safe.
|
|
if not PRIORITY_RE.search(new_content):
|
|
new_content = []
|
|
for line in content.split("\n"):
|
|
new_content.append(line)
|
|
if line.startswith('['):
|
|
new_content.append('priority=%d' % new_priority)
|
|
new_content = "\n".join(new_content)
|
|
return new_content
|
|
|
|
|
|
def _add_includepkgs(content):
|
|
new_content = []
|
|
for line in content.split("\n"):
|
|
new_content.append(line)
|
|
if line.startswith('['):
|
|
new_content.append(INCLUDE_PKGS)
|
|
return "\n".join(new_content)
|
|
|
|
|
|
def _inject_mirrors(content, args):
|
|
"""Replace any references to the default mirrors in repo content
|
|
|
|
In some cases we want to use mirrors whose repo files still point to the
|
|
default servers. If the user specified to use the mirror, we want to
|
|
replace any such references with the mirror address. This function
|
|
handles that by using a regex to swap out the baseurl server.
|
|
"""
|
|
|
|
content = re.sub('baseurl=%s' % DEFAULT_RDO_MIRROR,
|
|
'baseurl=%s' % args.rdo_mirror,
|
|
content)
|
|
|
|
if args.old_mirror:
|
|
content = re.sub('baseurl=%s' % args.old_mirror,
|
|
'baseurl=%s' % args.mirror,
|
|
content)
|
|
|
|
return content
|
|
|
|
|
|
def _install_repos(args, base_path):
|
|
def install_deps(args, base_path):
|
|
content = _get_repo(base_path + 'delorean-deps.repo', args)
|
|
_write_repo(content, args.output_path)
|
|
|
|
for repo in args.repos:
|
|
if repo == 'current':
|
|
content = _get_repo(base_path + 'current/delorean.repo', args)
|
|
_write_repo(content, args.output_path, name='delorean')
|
|
install_deps(args, base_path)
|
|
elif repo == 'deps':
|
|
install_deps(args, base_path)
|
|
elif repo == 'current-tripleo':
|
|
content = _get_repo(base_path + 'current-tripleo/delorean.repo',
|
|
args)
|
|
_write_repo(content, args.output_path)
|
|
install_deps(args, base_path)
|
|
elif repo == 'current-tripleo-dev':
|
|
content = _get_repo(base_path + 'delorean-deps.repo', args)
|
|
_write_repo(content, args.output_path)
|
|
content = _get_repo(base_path + 'current-tripleo/delorean.repo',
|
|
args)
|
|
content = TITLE_RE.sub('[\\1-current-tripleo]', content)
|
|
content = NAME_RE.sub('name=\\1-current-tripleo', content)
|
|
# We need to twiddle priorities since we're mixing multiple repos
|
|
# that are generated with the same priority.
|
|
content = _change_priority(content, 20)
|
|
_write_repo(content, args.output_path,
|
|
name='delorean-current-tripleo')
|
|
content = _get_repo(base_path + 'current/delorean.repo', args)
|
|
content = _add_includepkgs(content)
|
|
content = _change_priority(content, 10)
|
|
_write_repo(content, args.output_path, name='delorean')
|
|
elif repo == 'tripleo-ci-testing':
|
|
content = _get_repo(base_path + 'tripleo-ci-testing/delorean.repo',
|
|
args)
|
|
_write_repo(content, args.output_path)
|
|
install_deps(args, base_path)
|
|
elif repo == 'current-tripleo-rdo':
|
|
content = _get_repo(
|
|
base_path + 'current-tripleo-rdo/delorean.repo', args)
|
|
_write_repo(content, args.output_path)
|
|
install_deps(args, base_path)
|
|
elif repo == 'ceph':
|
|
if args.branch in ['liberty', 'mitaka']:
|
|
content = _create_ceph(args, 'hammer')
|
|
elif args.branch in ['newton', 'ocata', 'pike']:
|
|
content = _create_ceph(args, 'jewel')
|
|
elif args.branch in ['queens', 'rocky']:
|
|
content = _create_ceph(args, 'luminous')
|
|
elif args.branch in ['stein', 'train', 'ussuri', 'victoria']:
|
|
content = _create_ceph(args, 'nautilus')
|
|
else:
|
|
content = _create_ceph(args, 'pacific')
|
|
_write_repo(content, args.output_path)
|
|
elif repo == 'opstools':
|
|
content = OPSTOOLS_REPO_TEMPLATE % {'mirror': args.mirror}
|
|
_write_repo(content, args.output_path)
|
|
else:
|
|
raise InvalidArguments('Invalid repo "%s" specified' % repo)
|
|
|
|
distro = args.distro
|
|
# CentOS-8 AppStream is required for UBI-8
|
|
if distro == 'ubi8':
|
|
if not os.path.exists("/etc/distro.repos.d"):
|
|
print('WARNING: For UBI it is recommended to create '
|
|
'/etc/distro.repos.d and rerun!')
|
|
dp_exists = False
|
|
else:
|
|
dp_exists = True
|
|
if args.output_path == DEFAULT_OUTPUT_PATH and dp_exists:
|
|
distro_path = "/etc/distro.repos.d"
|
|
else:
|
|
distro_path = args.output_path
|
|
# TODO: Remove it once bugs are fixed
|
|
# Add extra options to APPSTREAM_REPO_TEMPLATE because of
|
|
# rhbz/1961558 and lpbz/1929634
|
|
extra = ''
|
|
if args.branch in ['train', 'ussuri', 'victoria']:
|
|
extra = 'exclude=edk2-ovmf-20200602gitca407c7246bf-5*'
|
|
content = APPSTREAM_REPO_TEMPLATE % {'mirror': args.mirror,
|
|
'extra': extra}
|
|
_write_repo(content, distro_path)
|
|
content = BASE_REPO_TEMPLATE % {'mirror': args.mirror}
|
|
_write_repo(content, distro_path)
|
|
distro = 'centos8' # switch it to continue as centos8 distro
|
|
|
|
# HA, Powertools are required for CentOS-8
|
|
if distro == 'centos8':
|
|
stream = '8'
|
|
if args.stream and not args.no_stream:
|
|
stream = stream + '-stream'
|
|
content = HIGHAVAILABILITY_REPO_TEMPLATE % {'mirror': args.mirror,
|
|
'stream': stream}
|
|
_write_repo(content, args.output_path)
|
|
content = POWERTOOLS_REPO_TEMPLATE % {'mirror': args.mirror,
|
|
'stream': stream}
|
|
_write_repo(content, args.output_path)
|
|
|
|
|
|
def _run_pkg_clean(distro):
|
|
pkg_mgr = 'yum' if distro == 'centos7' else 'dnf'
|
|
try:
|
|
subprocess.check_call([pkg_mgr, 'clean', 'metadata'])
|
|
except subprocess.CalledProcessError:
|
|
print('ERROR: Failed to clean yum metadata.')
|
|
raise
|
|
|
|
|
|
def main():
|
|
distro_id, distro_major_version_id, distro_name = _get_distro()
|
|
args = _parse_args(distro_id, distro_major_version_id)
|
|
_validate_args(args, distro_name)
|
|
base_path = _get_base_path(args)
|
|
if args.distro in ['centos7']:
|
|
_install_priorities()
|
|
_remove_existing(args)
|
|
_install_repos(args, base_path)
|
|
_run_pkg_clean(args.distro)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()
|