Add normalization for heat stacks

Change-Id: I564268d910a558288e661c15e279ca7fdfcc80f3
This commit is contained in:
Monty Taylor 2017-03-26 11:27:45 -05:00
parent c45b3f8902
commit 12523389a8
No known key found for this signature in database
GPG Key ID: 7BAE94BC7141A594
5 changed files with 111 additions and 15 deletions

View File

@ -451,3 +451,34 @@ A Magnum Service from magnum
report_count=int(), report_count=int(),
state=str(), state=str(),
properties=dict()) properties=dict())
Stack
-----
A Stack from Heat
.. code-block:: python
Stack = dict(
location=Location(),
id=str(),
name=str(),
created_at=str(),
deleted_at=str(),
updated_at=str(),
description=str(),
action=str(),
identifier=str(),
is_rollback_enabled=bool(),
notification_topics=list(),
outputs=list(),
owner=str(),
parameters=dict(),
parent=str(),
stack_user_project_id=str(),
status=str(),
status_reason=str(),
tags=dict(),
tempate_description=str(),
timeout_mins=int(),
properties=dict())

View File

@ -936,3 +936,62 @@ class Normalizer(object):
ret[key] = magnum_service.pop(key) ret[key] = magnum_service.pop(key)
ret['properties'] = magnum_service ret['properties'] = magnum_service
return ret return ret
def _normalize_stacks(self, stacks):
"""Normalize Heat Stacks"""
ret = []
for stack in stacks:
ret.append(self._normalize_stack(stack))
return ret
def _normalize_stack(self, stack):
"""Normalize Heat Stack"""
stack = stack.copy()
# Discard noise
stack.pop('HUMAN_ID', None)
stack.pop('human_id', None)
stack.pop('NAME_ATTR', None)
stack.pop('links', None)
# Discard things heatclient adds that aren't in the REST
stack.pop('action', None)
stack.pop('status', None)
stack.pop('identifier', None)
stack_status = stack.pop('stack_status')
(action, status) = stack_status.split('_')
ret = munch.Munch(
id=stack.pop('id'),
location=self._get_current_location(),
action=action,
status=status,
)
if not self.strict_mode:
ret['stack_status'] = stack_status
for (new_name, old_name) in (
('name', 'stack_name'),
('created_at', 'creation_time'),
('deleted_at', 'deletion_time'),
('updated_at', 'updated_time'),
('description', 'description'),
('is_rollback_enabled', 'disable_rollback'),
('parent', 'parent'),
('notification_topics', 'notification_topics'),
('parameters', 'parameters'),
('outputs', 'outputs'),
('owner', 'stack_owner'),
('status_reason', 'stack_status_reason'),
('stack_user_project_id', 'stack_user_project_id'),
('tempate_description', 'template_description'),
('timeout_mins', 'timeout_mins'),
('tags', 'tags')):
value = stack.pop(old_name, None)
ret[new_name] = value
if not self.strict_mode:
ret[old_name] = value
ret['identifier'] = '{name}/{id}'.format(
name=ret['name'], id=ret['id'])
ret['properties'] = stack
return ret

View File

@ -369,13 +369,6 @@ def normalize_flavor_accesses(flavor_accesses):
] ]
def normalize_stacks(stacks):
""" Normalize Stack Object """
for stack in stacks:
stack['name'] = stack['stack_name']
return stacks
def valid_kwargs(*valid_args): def valid_kwargs(*valid_args):
# This decorator checks if argument passed as **kwargs to a function are # This decorator checks if argument passed as **kwargs to a function are
# present in valid_args. # present in valid_args.

View File

@ -1771,7 +1771,7 @@ class OpenStackCloud(_normalize.Normalizer):
""" """
with _utils.shade_exceptions("Error fetching stack list"): with _utils.shade_exceptions("Error fetching stack list"):
stacks = self.manager.submit_task(_tasks.StackList()) stacks = self.manager.submit_task(_tasks.StackList())
return _utils.normalize_stacks(stacks) return self._normalize_stacks(stacks)
def list_server_security_groups(self, server): def list_server_security_groups(self, server):
"""List all security groups associated with the given server. """List all security groups associated with the given server.
@ -2781,7 +2781,7 @@ class OpenStackCloud(_normalize.Normalizer):
stacks = [stack] stacks = [stack]
except heat_exceptions.NotFound: except heat_exceptions.NotFound:
return [] return []
nstacks = _utils.normalize_stacks(stacks) nstacks = self._normalize_stacks(stacks)
return _utils._filter_list(nstacks, name_or_id, filters) return _utils._filter_list(nstacks, name_or_id, filters)
return _utils._get_entity( return _utils._get_entity(

View File

@ -33,7 +33,9 @@ class TestStack(base.TestCase):
mock_heat.stacks.list.return_value = fake_stacks mock_heat.stacks.list.return_value = fake_stacks
stacks = self.cloud.list_stacks() stacks = self.cloud.list_stacks()
mock_heat.stacks.list.assert_called_once_with() mock_heat.stacks.list.assert_called_once_with()
self.assertEqual(meta.obj_list_to_dict(fake_stacks), stacks) self.assertEqual(
self.cloud._normalize_stacks(meta.obj_list_to_dict(fake_stacks)),
stacks)
@mock.patch.object(shade.OpenStackCloud, 'heat_client') @mock.patch.object(shade.OpenStackCloud, 'heat_client')
def test_list_stacks_exception(self, mock_heat): def test_list_stacks_exception(self, mock_heat):
@ -53,19 +55,24 @@ class TestStack(base.TestCase):
mock_heat.stacks.list.return_value = fake_stacks mock_heat.stacks.list.return_value = fake_stacks
stacks = self.cloud.search_stacks() stacks = self.cloud.search_stacks()
mock_heat.stacks.list.assert_called_once_with() mock_heat.stacks.list.assert_called_once_with()
self.assertEqual(meta.obj_list_to_dict(fake_stacks), stacks) self.assertEqual(
self.cloud._normalize_stacks(meta.obj_list_to_dict(fake_stacks)),
stacks)
@mock.patch.object(shade.OpenStackCloud, 'heat_client') @mock.patch.object(shade.OpenStackCloud, 'heat_client')
def test_search_stacks_filters(self, mock_heat): def test_search_stacks_filters(self, mock_heat):
fake_stacks = [ fake_stacks = [
fakes.FakeStack('001', 'stack1', status='GOOD'), fakes.FakeStack('001', 'stack1', status='CREATE_COMPLETE'),
fakes.FakeStack('002', 'stack2', status='BAD'), fakes.FakeStack('002', 'stack2', status='CREATE_FAILED'),
] ]
mock_heat.stacks.list.return_value = fake_stacks mock_heat.stacks.list.return_value = fake_stacks
filters = {'stack_status': 'GOOD'} filters = {'status': 'COMPLETE'}
stacks = self.cloud.search_stacks(filters=filters) stacks = self.cloud.search_stacks(filters=filters)
mock_heat.stacks.list.assert_called_once_with() mock_heat.stacks.list.assert_called_once_with()
self.assertEqual(meta.obj_list_to_dict(fake_stacks[:1]), stacks) self.assertEqual(
self.cloud._normalize_stacks(
meta.obj_list_to_dict(fake_stacks[:1])),
stacks)
@mock.patch.object(shade.OpenStackCloud, 'heat_client') @mock.patch.object(shade.OpenStackCloud, 'heat_client')
def test_search_stacks_exception(self, mock_heat): def test_search_stacks_exception(self, mock_heat):
@ -138,6 +145,8 @@ class TestStack(base.TestCase):
@mock.patch.object(shade.OpenStackCloud, 'heat_client') @mock.patch.object(shade.OpenStackCloud, 'heat_client')
def test_create_stack(self, mock_heat, mock_template): def test_create_stack(self, mock_heat, mock_template):
mock_template.return_value = ({}, {}) mock_template.return_value = ({}, {})
mock_heat.stacks.create.return_value = fakes.FakeStack('001', 'stack1')
mock_heat.stacks.get.return_value = fakes.FakeStack('001', 'stack1')
self.cloud.create_stack('stack_name') self.cloud.create_stack('stack_name')
self.assertTrue(mock_template.called) self.assertTrue(mock_template.called)
mock_heat.stacks.create.assert_called_once_with( mock_heat.stacks.create.assert_called_once_with(
@ -159,6 +168,8 @@ class TestStack(base.TestCase):
stack = {'id': 'stack_id', 'name': 'stack_name'} stack = {'id': 'stack_id', 'name': 'stack_name'}
mock_template.return_value = ({}, {}) mock_template.return_value = ({}, {})
mock_get.return_value = stack mock_get.return_value = stack
mock_heat.stacks.create.return_value = fakes.FakeStack('001', 'stack1')
mock_heat.stacks.get.return_value = fakes.FakeStack('001', 'stack1')
ret = self.cloud.create_stack('stack_name', wait=True) ret = self.cloud.create_stack('stack_name', wait=True)
self.assertTrue(mock_template.called) self.assertTrue(mock_template.called)
mock_heat.stacks.create.assert_called_once_with( mock_heat.stacks.create.assert_called_once_with(
@ -178,6 +189,8 @@ class TestStack(base.TestCase):
@mock.patch.object(shade.OpenStackCloud, 'heat_client') @mock.patch.object(shade.OpenStackCloud, 'heat_client')
def test_update_stack(self, mock_heat, mock_template): def test_update_stack(self, mock_heat, mock_template):
mock_template.return_value = ({}, {}) mock_template.return_value = ({}, {})
mock_heat.stacks.update.return_value = fakes.FakeStack('001', 'stack1')
mock_heat.stacks.get.return_value = fakes.FakeStack('001', 'stack1')
self.cloud.update_stack('stack_name') self.cloud.update_stack('stack_name')
self.assertTrue(mock_template.called) self.assertTrue(mock_template.called)
mock_heat.stacks.update.assert_called_once_with( mock_heat.stacks.update.assert_called_once_with(