Merge "Automatically set LIBS_FROM_GIT based on required projects"

This commit is contained in:
Zuul 2018-04-19 12:44:31 +00:00 committed by Gerrit Code Review
commit a0f319fc6d
6 changed files with 110 additions and 34 deletions

View File

@ -429,22 +429,6 @@ function lib_installed_from_git {
[[ -n $(pip list --format=columns 2>/dev/null | awk "/^$safe_name/ {print \$3}") ]] [[ -n $(pip list --format=columns 2>/dev/null | awk "/^$safe_name/ {print \$3}") ]]
} }
# check that everything that's in LIBS_FROM_GIT was actually installed
# correctly, this helps double check issues with library fat fingering.
function check_libs_from_git {
local lib=""
local not_installed=""
for lib in $(echo ${LIBS_FROM_GIT} | tr "," " "); do
if ! lib_installed_from_git "$lib"; then
not_installed+=" $lib"
fi
done
# if anything is not installed, say what it is.
if [[ -n "$not_installed" ]]; then
die $LINENO "The following LIBS_FROM_GIT were not installed correct: $not_installed"
fi
}
# setup a library by name. If we are trying to use the library from # setup a library by name. If we are trying to use the library from
# git, we'll do a git based install, otherwise we'll punt and the # git, we'll do a git based install, otherwise we'll punt and the
# library should be installed by a requirements pull from another # library should be installed by a requirements pull from another
@ -555,6 +539,13 @@ function _setup_package_with_constraints_edit {
setup_package $project_dir "$flags" $extras setup_package $project_dir "$flags" $extras
# If this project is in LIBS_FROM_GIT, verify it was actually installed
# correctly. This helps catch errors caused by constraints mismatches.
if use_library_from_git "$project_dir"; then
if ! lib_installed_from_git "$project_dir"; then
die $LINENO "The following LIBS_FROM_GIT was not installed correctly: $project_dir"
fi
fi
} }
# ``pip install -e`` the package, which processes the dependencies # ``pip install -e`` the package, which processes the dependencies

View File

@ -20,6 +20,14 @@ Write the local.conf file for use by devstack
bash shell variables, and will be ordered so that variables used by bash shell variables, and will be ordered so that variables used by
later entries appear first. later entries appear first.
As a special case, the variable ``LIBS_FROM_GIT`` will be
constructed automatically from the projects which appear in the
``required-projects`` list defined by the job. To instruct
devstack to install a library from source rather than pypi, simply
add that library to the job's ``required-projects`` list. To
override the automatically-generated value, set ``LIBS_FROM_GIT``
in ``devstack_localrc`` to the desired value.
.. zuul:rolevar:: devstack_local_conf .. zuul:rolevar:: devstack_local_conf
:type: dict :type: dict
@ -75,3 +83,7 @@ Write the local.conf file for use by devstack
A dictionary mapping a plugin name to a git repo location. If the A dictionary mapping a plugin name to a git repo location. If the
location is a non-empty string, then an ``enable_plugin`` line will location is a non-empty string, then an ``enable_plugin`` line will
be emmitted for the plugin name. be emmitted for the plugin name.
If a plugin declares a dependency on another plugin (via
``plugin_requires`` in the plugin's settings file), this role will
automatically emit ``enable_plugin`` lines in the correct order.

View File

@ -207,16 +207,16 @@ class PluginGraph(DependencyGraph):
class LocalConf(object): class LocalConf(object):
def __init__(self, localrc, localconf, base_services, services, plugins, def __init__(self, localrc, localconf, base_services, services, plugins,
base_dir): base_dir, projects):
self.localrc = [] self.localrc = []
self.meta_sections = {} self.meta_sections = {}
self.plugin_deps = {} self.plugin_deps = {}
self.base_dir = base_dir self.base_dir = base_dir
self.projects = projects
if plugins: if plugins:
self.handle_plugins(plugins) self.handle_plugins(plugins)
if services or base_services: if services or base_services:
self.handle_services(base_services, services or {}) self.handle_services(base_services, services or {})
if localrc:
self.handle_localrc(localrc) self.handle_localrc(localrc)
if localconf: if localconf:
self.handle_localconf(localconf) self.handle_localconf(localconf)
@ -241,9 +241,22 @@ class LocalConf(object):
self.localrc.append('enable_service {}'.format(k)) self.localrc.append('enable_service {}'.format(k))
def handle_localrc(self, localrc): def handle_localrc(self, localrc):
lfg = False
if localrc:
vg = VarGraph(localrc) vg = VarGraph(localrc)
for k, v in vg.getVars(): for k, v in vg.getVars():
self.localrc.append('{}={}'.format(k, v)) self.localrc.append('{}={}'.format(k, v))
if k == 'LIBS_FROM_GIT':
lfg = True
if not lfg and self.projects:
required_projects = []
for project_name, project_info in self.projects.items():
if project_info.get('required'):
required_projects.append(project_info['short_name'])
if required_projects:
self.localrc.append('LIBS_FROM_GIT={}'.format(
','.join(required_projects)))
def handle_localconf(self, localconf): def handle_localconf(self, localconf):
for phase, phase_data in localconf.items(): for phase, phase_data in localconf.items():
@ -277,6 +290,7 @@ def main():
local_conf=dict(type='dict'), local_conf=dict(type='dict'),
base_dir=dict(type='path'), base_dir=dict(type='path'),
path=dict(type='str'), path=dict(type='str'),
projects=dict(type='dict'),
) )
) )
@ -286,7 +300,8 @@ def main():
p.get('base_services'), p.get('base_services'),
p.get('services'), p.get('services'),
p.get('plugins'), p.get('plugins'),
p.get('base_dir')) p.get('base_dir'),
p.get('projects'))
lc.write(p['path']) lc.write(p['path'])
module.exit_json() module.exit_json()

View File

@ -56,7 +56,8 @@ class TestDevstackLocalConf(unittest.TestCase):
p.get('base_services'), p.get('base_services'),
p.get('services'), p.get('services'),
p.get('plugins'), p.get('plugins'),
p.get('base_dir')) p.get('base_dir'),
p.get('projects'))
lc.write(p['path']) lc.write(p['path'])
plugins = [] plugins = []
@ -66,6 +67,7 @@ class TestDevstackLocalConf(unittest.TestCase):
plugins.append(line.split()[1]) plugins.append(line.split()[1])
self.assertEqual(['bar', 'baz', 'foo'], plugins) self.assertEqual(['bar', 'baz', 'foo'], plugins)
def test_plugin_deps(self): def test_plugin_deps(self):
"Test that plugins with dependencies work" "Test that plugins with dependencies work"
os.makedirs(os.path.join(self.tmpdir, 'foo-plugin', 'devstack')) os.makedirs(os.path.join(self.tmpdir, 'foo-plugin', 'devstack'))
@ -101,20 +103,80 @@ class TestDevstackLocalConf(unittest.TestCase):
plugins=plugins, plugins=plugins,
base_dir=self.tmpdir, base_dir=self.tmpdir,
path=os.path.join(self.tmpdir, 'test.local.conf')) path=os.path.join(self.tmpdir, 'test.local.conf'))
def test_libs_from_git(self):
"Test that LIBS_FROM_GIT is auto-generated"
projects = {
'git.openstack.org/openstack/nova': {
'required': True,
'short_name': 'nova',
},
'git.openstack.org/openstack/oslo.messaging': {
'required': True,
'short_name': 'oslo.messaging',
},
'git.openstack.org/openstack/devstack-plugin': {
'required': False,
'short_name': 'devstack-plugin',
},
}
p = dict(base_services=[],
base_dir='./test',
path=os.path.join(self.tmpdir, 'test.local.conf'),
projects=projects)
lc = LocalConf(p.get('localrc'), lc = LocalConf(p.get('localrc'),
p.get('local_conf'), p.get('local_conf'),
p.get('base_services'), p.get('base_services'),
p.get('services'), p.get('services'),
p.get('plugins'), p.get('plugins'),
p.get('base_dir')) p.get('base_dir'),
p.get('projects'))
lc.write(p['path']) lc.write(p['path'])
plugins = [] lfg = None
with open(p['path']) as f: with open(p['path']) as f:
for line in f: for line in f:
if line.startswith('enable_plugin'): if line.startswith('LIBS_FROM_GIT'):
plugins.append(line.split()[1]) lfg = line.strip().split('=')[1]
self.assertEqual(['foo', 'bar'], plugins) self.assertEqual('nova,oslo.messaging', lfg)
def test_overridelibs_from_git(self):
"Test that LIBS_FROM_GIT can be overridden"
localrc = {'LIBS_FROM_GIT': 'oslo.db'}
projects = {
'git.openstack.org/openstack/nova': {
'required': True,
'short_name': 'nova',
},
'git.openstack.org/openstack/oslo.messaging': {
'required': True,
'short_name': 'oslo.messaging',
},
'git.openstack.org/openstack/devstack-plugin': {
'required': False,
'short_name': 'devstack-plugin',
},
}
p = dict(localrc=localrc,
base_services=[],
base_dir='./test',
path=os.path.join(self.tmpdir, 'test.local.conf'),
projects=projects)
lc = LocalConf(p.get('localrc'),
p.get('local_conf'),
p.get('base_services'),
p.get('services'),
p.get('plugins'),
p.get('base_dir'),
p.get('projects'))
lc.write(p['path'])
lfg = None
with open(p['path']) as f:
for line in f:
if line.startswith('LIBS_FROM_GIT'):
lfg = line.strip().split('=')[1]
self.assertEqual('oslo.db', lfg)
def test_plugin_circular_deps(self): def test_plugin_circular_deps(self):
"Test that plugins with circular dependencies fail" "Test that plugins with circular dependencies fail"

View File

@ -9,3 +9,4 @@
localrc: "{{ devstack_localrc|default(omit) }}" localrc: "{{ devstack_localrc|default(omit) }}"
local_conf: "{{ devstack_local_conf|default(omit) }}" local_conf: "{{ devstack_local_conf|default(omit) }}"
base_dir: "{{ devstack_base_dir|default(omit) }}" base_dir: "{{ devstack_base_dir|default(omit) }}"
projects: "{{ zuul.projects }}"

View File

@ -1398,11 +1398,6 @@ fi
# Check the status of running services # Check the status of running services
service_check service_check
# ensure that all the libraries we think we installed from git,
# actually were.
check_libs_from_git
# Configure nova cellsv2 # Configure nova cellsv2
# ---------------------- # ----------------------