tripleo-common/tripleo_common/actions/heat_capabilities.py
Dougal Matthews 9d3d03436a Remove the underscore from the base action client methods
The underscore typically designates these methods as private, but
they are intended to be used by any/all of the subclasses. This
change clarifies the API.

Change-Id: I4dc95a0e1f84570b184f8b75f51a2c1456909fb9
2016-11-28 13:09:38 +00:00

196 lines
7.0 KiB
Python

# Copyright 2106 Red Hat, Inc.
# 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.
import fnmatch
import logging
import yaml
from mistral.workflow import utils as mistral_workflow_utils
from tripleo_common.actions import base
from tripleo_common import constants
LOG = logging.getLogger(__name__)
class GetCapabilitiesAction(base.TripleOAction):
"""Gets list of available heat environments
Parses the capabilities_map.yaml file in a given plan and
returns a list of environments
:param container: name of the swift container / plan name
:return: list of environment files in swift container
"""
def __init__(self, container=constants.DEFAULT_CONTAINER_NAME):
super(GetCapabilitiesAction, self).__init__()
self.container = container
def run(self):
try:
swift_client = self.get_object_client()
map_file = swift_client.get_object(
self.container, 'capabilities-map.yaml')
capabilities = yaml.safe_load(map_file[1])
except Exception:
err_msg = (
"Error parsing capabilities-map.yaml.")
LOG.exception(err_msg)
return mistral_workflow_utils.Result(error=err_msg)
try:
container_files = swift_client.get_container(self.container)
container_file_list = [entry['name'] for entry
in container_files[1]]
except Exception as swift_err:
err_msg = ("Error retrieving plan files: %s" % swift_err)
LOG.exception(err_msg)
return mistral_workflow_utils.Result(error=err_msg)
try:
mistral_client = self.get_workflow_client()
mistral_env = mistral_client.environments.get(self.container)
except Exception as mistral_err:
err_msg = ("Error retrieving mistral "
"environment. %s" % mistral_err)
LOG.exception(err_msg)
return mistral_workflow_utils.Result(error=err_msg)
selected_envs = [item['path'] for item in
mistral_env.variables['environments']
if 'path' in item]
# extract environment files
plan_environments = []
for env_group in capabilities['topics']:
for envs in env_group['environment_groups']:
for files in envs['environments']:
file = files.get('file')
if file:
plan_environments.append(file)
# parse plan for environment files
env_files = fnmatch.filter(
container_file_list, '*environments/*.yaml')
env_user_files = fnmatch.filter(
container_file_list, '*user-environment.yaml')
outstanding_envs = list(set(env_files).union(
env_user_files) - set(plan_environments))
# change capabilities format
data_to_return = {}
capabilities.pop('root_environment')
capabilities.pop('root_template')
for topic in capabilities['topics']:
title = topic.get('title', '_title_holder')
data_to_return[title] = topic
for eg in topic['environment_groups']:
for env in eg['environments']:
if selected_envs and env.get('file') in selected_envs:
env['enabled'] = True
else:
env['enabled'] = False
# add custom environment files
other_environments = []
for env in outstanding_envs:
flag = selected_envs and env in selected_envs
new_env = {
"description": "Enable %s environment" % env,
"enabled": flag,
"file": env,
"title": env,
}
other_environments.append(new_env)
other_environments.sort(key=lambda x: x['file'])
other_environment_groups = []
for group in other_environments:
new_group = {
"description": None,
"environments": [group],
"title": group['file'],
}
other_environment_groups.append(new_group)
other_environments_topic_dict = {
"description": None,
"title": "Other",
"environment_groups": other_environment_groups
}
other_environments_topic = {
"Other": other_environments_topic_dict
}
data_to_return.update(other_environments_topic)
return data_to_return
class UpdateCapabilitiesAction(base.TripleOAction):
"""Updates Mistral Environment with selected environments
Takes a list of environment files and depending on the value of the
enabled flag, adds or removes them from the Mistral Environment.
:param environments: list of environments
:param container: name of the swift container / plan name
:return: the updated mistral environment
"""
def __init__(self, environments,
container=constants.DEFAULT_CONTAINER_NAME):
super(UpdateCapabilitiesAction, self).__init__()
self.container = container
self.environments = environments
def run(self):
mistral_client = self.get_workflow_client()
mistral_env = None
try:
mistral_env = mistral_client.environments.get(self.container)
except Exception as mistral_err:
err_msg = (
"Error retrieving mistral "
"environment. %s" % mistral_err)
LOG.exception(err_msg)
return mistral_workflow_utils.Result(error=err_msg)
for k, v in self.environments.items():
found = False
if {'path': k} in mistral_env.variables['environments']:
found = True
if v:
if not found:
mistral_env.variables['environments'].append(
{'path': k}
)
else:
if found:
mistral_env.variables['environments'].remove({'path': k})
env_kwargs = {
'name': mistral_env.name,
'variables': mistral_env.variables
}
try:
mistral_client.environments.update(**env_kwargs)
except Exception as mistral_err:
err_msg = (
"Error retrieving mistral environment. %s" % mistral_err)
LOG.exception(err_msg)
return mistral_workflow_utils.Result(error=err_msg)
return mistral_env.variables