Filter out disabled services from prepare command

Previously, the `openstack overcloud container image prepare` looked
only at the resource registry -- checking if a resource pointed to
a containerized service -- to determine which images to use without
looking at the list of services that are enabled or not per roles.

This commit introduces a new '--roles-file' option to the  command to
specify the path to the roles file when not using the default.

Closes-Bug: #1714296
Change-Id: I0188df062bd341dcf4791bfa9c9d0f1ad7f9396e
This commit is contained in:
Martin André 2017-08-31 17:38:24 +02:00
parent 5ebe9b18ed
commit 954ed90435
2 changed files with 75 additions and 7 deletions

View File

@ -74,6 +74,43 @@ class TestContainerImagePrepare(TestPluginV1):
# Get the command object to test # Get the command object to test
self.cmd = container_image.PrepareImageFiles(self.app, None) self.cmd = container_image.PrepareImageFiles(self.app, None)
def test_get_enabled_services(self):
temp = tempfile.mkdtemp()
self.addCleanup(shutil.rmtree, temp)
resource_registry = {'parameter_defaults': {
'RoleDisabledViaEnvironmentCount': 0,
'RoleOverwrittenViaEnvironmentServices': [
'OS::TripleO::Services::FromResourceRegistry'
]
}}
roles_file = '/foo/roles_data.yaml'
roles_yaml = '''
- name: EnabledRole
CountDefault: 1
ServicesDefault:
- OS::TripleO::Services::AodhEvaluator
- name: RoleDisabledViaRolesData
CountDefault: 0
ServicesDefault:
- OS::TripleO::Services::AodhApi
- name: RoleDisabledViaEnvironment
CountDefault: 1
ServicesDefault:
- OS::TripleO::Services::Disabled
- name: RoleOverwrittenViaEnvironment
CountDefault: 1
ServicesDefault:
- OS::TripleO::Services::Overwritten
'''
mock_open_context = mock.mock_open(read_data=roles_yaml)
with mock.patch('six.moves.builtins.open', mock_open_context):
enabled_services = self.cmd.get_enabled_services(resource_registry,
roles_file)
mock_open_context.assert_called_once_with(roles_file)
self.assertEqual(set(['OS::TripleO::Services::AodhEvaluator',
'OS::TripleO::Services::FromResourceRegistry']),
enabled_services)
@mock.patch('tripleo_common.image.kolla_builder.KollaImageBuilder') @mock.patch('tripleo_common.image.kolla_builder.KollaImageBuilder')
def test_container_image_prepare_noargs(self, mock_builder): def test_container_image_prepare_noargs(self, mock_builder):
arglist = [] arglist = []
@ -106,8 +143,7 @@ class TestContainerImagePrepare(TestPluginV1):
images_file = os.path.join(temp, 'overcloud_containers.yaml') images_file = os.path.join(temp, 'overcloud_containers.yaml')
env_file = os.path.join(temp, 'containers_env.yaml') env_file = os.path.join(temp, 'containers_env.yaml')
tmpl_file = os.path.join(temp, 'overcloud_containers.yaml.j2') tmpl_file = os.path.join(temp, 'overcloud_containers.yaml.j2')
aodh_file = os.path.join(temp, 'docker', 'services', aodh_file = os.path.join(temp, 'docker', 'services', 'aodh.yaml')
'overcloud_containers.yaml.j2')
resource_registry = {'resource_registry': { resource_registry = {'resource_registry': {
'OS::TripleO::Services::AodhEvaluator': aodh_file, 'OS::TripleO::Services::AodhEvaluator': aodh_file,

View File

@ -30,6 +30,8 @@ import yaml
from tripleo_common.image import image_uploader from tripleo_common.image import image_uploader
from tripleo_common.image import kolla_builder from tripleo_common.image import kolla_builder
from tripleoclient import constants
class UploadImage(command.Command): class UploadImage(command.Command):
"""Push overcloud container images to registries.""" """Push overcloud container images to registries."""
@ -163,6 +165,8 @@ class PrepareImageFiles(command.Command):
template_file = os.path.join(sys.prefix, 'share', 'tripleo-common', template_file = os.path.join(sys.prefix, 'share', 'tripleo-common',
'container-images', 'container-images',
'overcloud_containers.yaml.j2') 'overcloud_containers.yaml.j2')
roles_file = os.path.join(constants.TRIPLEO_HEAT_TEMPLATES,
constants.OVERCLOUD_ROLES_FILE)
parser.add_argument( parser.add_argument(
"--template-file", "--template-file",
dest="template_file", dest="template_file",
@ -258,6 +262,12 @@ class PrepareImageFiles(command.Command):
help=_("File to write heat environment file which specifies all " help=_("File to write heat environment file which specifies all "
"image parameters. Any existing file will be overwritten."), "image parameters. Any existing file will be overwritten."),
) )
parser.add_argument(
'--roles-file', '-r', dest='roles_file',
default=roles_file,
help=_('Roles file, overrides the default %s'
) % constants.OVERCLOUD_ROLES_FILE
)
return parser return parser
def parse_set_values(self, subs, set_values): def parse_set_values(self, subs, set_values):
@ -285,20 +295,42 @@ class PrepareImageFiles(command.Command):
yaml.safe_dump({'parameter_defaults': params}, f, yaml.safe_dump({'parameter_defaults': params}, f,
default_flow_style=False) default_flow_style=False)
def build_service_filter(self, environment_files): def get_enabled_services(self, environment, roles_file):
enabled_services = set()
try:
roles_data = yaml.safe_load(open(roles_file).read())
except IOError:
return enabled_services
parameter_defaults = environment.get('parameter_defaults', {})
for role in roles_data:
count = parameter_defaults.get('%sCount' % role['name'],
role.get('CountDefault', 0))
if count > 0:
enabled_services.update(
parameter_defaults.get('%sServices' % role['name'],
role.get('ServicesDefault', [])))
return enabled_services
def build_service_filter(self, environment_files, roles_file):
# Do not filter unless asked for it
if not environment_files: if not environment_files:
return None return None
service_filter = set()
env_files, env = ( env_files, env = (
template_utils.process_multiple_environments_and_files( template_utils.process_multiple_environments_and_files(
environment_files)) environment_files))
enabled_services = self.get_enabled_services(env, roles_file)
containerized_services = set()
for service, env_path in env.get('resource_registry', {}).items(): for service, env_path in env.get('resource_registry', {}).items():
# Use the template path to determine if it represents a # Use the template path to determine if it represents a
# containerized service # containerized service
if '/docker/services/' in env_path: if '/docker/services/' in env_path:
service_filter.add(service) containerized_services.add(service)
return service_filter
return containerized_services.intersection(enabled_services)
def take_action(self, parsed_args): def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args) self.log.debug("take_action(%s)" % parsed_args)
@ -311,7 +343,7 @@ class PrepareImageFiles(command.Command):
self.parse_set_values(subs, parsed_args.set) self.parse_set_values(subs, parsed_args.set)
service_filter = self.build_service_filter( service_filter = self.build_service_filter(
parsed_args.environment_files) parsed_args.environment_files, parsed_args.roles_file)
def ffunc(entry): def ffunc(entry):
imagename = entry.get('imagename', '') imagename = entry.get('imagename', '')