Add networking-sfc port-pair-group resource plug-in

This patch adds OS::Neutron::PortpairGroup sfc resource

blueprint : sfc-heat

Change-Id: I8ad274e697b2a950449172a5a3e71a8f7b9fd7c9
This commit is contained in:
mohankumar_n 2016-07-16 00:11:53 +05:30
parent 5c74723f5e
commit 0dbe25a2b1
5 changed files with 360 additions and 12 deletions

View File

@ -17,6 +17,7 @@ from neutronclient.v2_0 import client as nc
from oslo_utils import uuidutils
from heat.common import exception
from heat.common.i18n import _
from heat.engine.clients import client_plugin
from heat.engine.clients import os as os_client
@ -155,17 +156,20 @@ class NeutronClientPlugin(client_plugin.ClientPlugin):
raise exception.PhysicalResourceNameAmbiguity(name=sg)
return seclist
def _resove_resource_path(self, resource):
def _resolve_resource_path(self, resource):
"""Returns sfc resource path."""
if resource == 'port_pair':
RESOURCE_PATH = "/sfc/port_pairs"
return RESOURCE_PATH
path = "/sfc/port_pairs"
elif resource == 'port_pair_group':
path = "/sfc/port_pair_groups"
return path
def create_sfc_resource(self, resource, props):
"""Returns created sfc resource record."""
path = self._resove_resource_path(resource)
path = self._resolve_resource_path(resource)
record = self.client().create_ext(path, {resource: props}
).get(resource)
return record
@ -173,19 +177,48 @@ class NeutronClientPlugin(client_plugin.ClientPlugin):
def update_sfc_resource(self, resource, prop_diff, resource_id):
"""Returns updated sfc resource record."""
path = self._resove_resource_path(resource)
return self.client().update_ext(path + '/%s', self.resource_id,
path = self._resolve_resource_path(resource)
return self.client().update_ext(path + '/%s', resource_id,
{resource: prop_diff})
def delete_sfc_resource(self, resource, resource_id):
"""deletes sfc resource record and returns status"""
"""Deletes sfc resource record and returns status."""
path = self._resove_resource_path(resource)
return self.client().delete_ext(path + '/%s', self.resource_id)
path = self._resolve_resource_path(resource)
return self.client().delete_ext(path + '/%s', resource_id)
def show_sfc_resource(self, resource, resource_id):
"""returns specific sfc resource record"""
"""Returns specific sfc resource record."""
path = self._resove_resource_path(resource)
return self.client().show_ext(path + '/%s', self.resource_id
path = self._resolve_resource_path(resource)
return self.client().show_ext(path + '/%s', resource_id
).get(resource)
def resolve_ext_resource(self, resource, name_or_id):
"""Returns the id and validate neutron ext resource."""
path = self._resolve_resource_path(resource)
try:
record = self.client().show_ext(path + '/%s', name_or_id)
return record.get(resource).get('id')
except exceptions.NotFound:
res_plural = resource + 's'
result = self.client().list_ext(collection=res_plural,
path=path, retrieve_all=True)
resources = result.get(res_plural)
matched = []
for res in resources:
if res.get('name') == name_or_id:
matched.append(res.get('id'))
if len(matched) > 1:
raise exceptions.NeutronClientNoUniqueMatch(resource=resource,
name=name_or_id)
elif len(matched) == 0:
not_found_message = (_("Unable to find %(resource)s with name "
"or id '%(name_or_id)s'") %
{'resource': resource,
'name_or_id': name_or_id})
raise exceptions.NotFound(message=not_found_message)
else:
return matched[0]

View File

@ -59,6 +59,17 @@ class NeutronConstraint(constraints.BaseCustomConstraint):
self.resource_name, value, cmd_resource=self.cmd_resource)
class NeutronExtConstraint(NeutronConstraint):
def validate_with_client(self, client, value):
neutron_plugin = client.client_plugin(CLIENT_NAME)
if (self.extension and
not neutron_plugin.has_extension(self.extension)):
raise exception.EntityNotFound(entity='neutron extension',
name=self.extension)
neutron_plugin.resolve_ext_resource(self.resource_name, value)
class PortConstraint(NeutronConstraint):
resource_name = 'port'
@ -90,6 +101,11 @@ class QoSPolicyConstraint(NeutronConstraint):
extension = 'qos'
class PortPairConstraint(NeutronExtConstraint):
resource_name = 'port_pair'
extension = 'sfc'
class ProviderConstraint(constraints.BaseCustomConstraint):
expected_exceptions = (exception.StackValidationFailed,)

View File

@ -0,0 +1,112 @@
#
# 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 heat.common.i18n import _
from heat.engine import constraints
from heat.engine import properties
from heat.engine.resources.openstack.neutron import neutron
from heat.engine import support
from heat.engine import translation
class PortPairGroup(neutron.NeutronResource):
"""Heat Template Resource for networking-sfc port-pair-group.
Multiple port-pairs may be included in a port-pair-group to allow the
specification of a set of functionally equivalent Service Functions that
can be be used for load distribution.
"""
support_status = support.SupportStatus(
version='8.0.0',
status=support.UNSUPPORTED)
required_service_extension = 'sfc'
PROPERTIES = (
NAME, DESCRIPTION, PORT_PAIRS,
) = (
'name', 'description', 'port_pairs',
)
properties_schema = {
NAME: properties.Schema(
properties.Schema.STRING,
_('Name for the Port Pair Group.'),
update_allowed=True
),
DESCRIPTION: properties.Schema(
properties.Schema.STRING,
_('Description for the Port Pair Group.'),
update_allowed=True
),
PORT_PAIRS: properties.Schema(
properties.Schema.LIST,
_('A list of Port Pair IDs or names to apply.'),
required=True,
update_allowed=True,
schema=properties.Schema(
properties.Schema.STRING,
_('Port Pair ID or name .'),
constraints=[
constraints.CustomConstraint('neutron.port_pair')
]
)
),
}
def translation_rules(self, props):
return [
translation.TranslationRule(
props,
translation.TranslationRule.RESOLVE,
[self.PORT_PAIRS],
client_plugin=self.client_plugin(),
finder='resolve_ext_resource',
entity='port_pair'
)
]
def _show_resource(self):
return self.client_plugin().show_sfc_resource('port_pair_group',
self.resource_id)
def handle_create(self):
props = self.prepare_properties(
self.properties,
self.physical_resource_name())
port_pair_group = self.client_plugin().create_sfc_resource(
'port_pair_group', props)
self.resource_id_set(port_pair_group['id'])
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
if prop_diff:
self.prepare_update_properties(prop_diff)
self.client_plugin().update_sfc_resource(
'port_pair_group',
prop_diff,
self.resource_id)
def handle_delete(self):
if self.resource_id is None:
return
with self.client_plugin().ignore_not_found:
self.client_plugin().delete_sfc_resource('port_pair_group',
self.resource_id)
def resource_mapping():
return {
'OS::Neutron::PortPairGroup': PortPairGroup,
}

View File

@ -0,0 +1,186 @@
#
# 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 mock
from heat.engine.clients.os import neutron
from heat.engine.resources.openstack.neutron.sfc import port_pair_group
from heat.engine import stack
from heat.engine import template
from heat.tests import common
from heat.tests import utils
sample_template = {
'heat_template_version': '2016-04-08',
'resources': {
'test_resource': {
'type': 'OS::Neutron::PortPairGroup',
'properties': {
'name': 'test_port_pair_group',
'description': 'desc',
'port_pairs': ['port1']
}
}
}
}
RESOURCE_TYPE = 'OS::Neutron::PortPairGroup'
class PortPairGroupTest(common.HeatTestCase):
def setUp(self):
super(PortPairGroupTest, self).setUp()
self.patchobject(neutron.NeutronClientPlugin, 'has_extension',
return_value=True)
self.ctx = utils.dummy_context()
self.stack = stack.Stack(
self.ctx, 'test_stack',
template.Template(sample_template)
)
self.test_resource = self.stack['test_resource']
self.test_client_plugin = mock.MagicMock()
self.test_resource.client_plugin = mock.MagicMock(
return_value=self.test_client_plugin)
self.test_client = mock.MagicMock()
self.test_resource.client = mock.MagicMock(
return_value=self.test_client)
self.test_client_plugin.get_notification = mock.MagicMock(
return_value='sample_notification')
self.patchobject(self.test_client_plugin,
'resolve_ext_resource').return_value = ('port1')
def test_resource_mapping(self):
mapping = port_pair_group.resource_mapping()
self.assertEqual(port_pair_group.PortPairGroup,
mapping['OS::Neutron::PortPairGroup'])
def _get_mock_resource(self):
value = mock.MagicMock()
value.id = '477e8273-60a7-4c41-b683-fdb0bc7cd152'
return value
def _resolve_sfc_resource(self):
value = mock.MagicMock()
value.id = '[port1]'
return value.id
def test_resource_handle_create(self):
mock_ppg_create = self.test_client_plugin.create_sfc_resource
mock_resource = self._get_mock_resource()
mock_ppg_create.return_value = mock_resource
# validate the properties
self.assertEqual(
'test_port_pair_group',
self.test_resource.properties.get(
port_pair_group.PortPairGroup.NAME))
self.assertEqual(
'desc',
self.test_resource.properties.get(
port_pair_group.PortPairGroup.DESCRIPTION))
self.assertEqual(
['port1'],
self.test_resource.properties.get(
port_pair_group.PortPairGroup.PORT_PAIRS))
self.test_resource.data_set = mock.Mock()
self.test_resource.handle_create()
mock_ppg_create.assert_called_once_with(
'port_pair_group',
{
'name': 'test_port_pair_group',
'description': 'desc',
'port_pairs': ['port1'],
}
)
def test_resource_handle_delete(self):
mock_ppg_delete = self.test_client_plugin.delete_sfc_resource
self.test_resource.resource_id = '477e8273-60a7-4c41-b683-fdb0bc7cd151'
mock_ppg_delete.return_value = None
self.assertIsNone(self.test_resource.handle_delete())
mock_ppg_delete.assert_called_once_with(
'port_pair_group', self.test_resource.resource_id)
def test_resource_handle_delete_resource_id_is_none(self):
self.test_resource.resource_id = None
self.assertIsNone(self.test_resource.handle_delete())
self.assertEqual(0, self.test_client_plugin.
delete_sfc_resource.call_count)
def test_resource_handle_delete_not_found(self):
self.test_resource.resource_id = '477e8273-60a7-4c41-b683-fdb0bc7cd151'
mock_ppg_delete = self.test_client_plugin.delete_sfc_resource
mock_ppg_delete.side_effect = self.test_client_plugin.NotFound
self.assertIsNone(self.test_resource.handle_delete())
def test_resource_show_resource(self):
mock_ppg_get = self.test_client_plugin.show_sfc_resource
mock_ppg_get.return_value = {}
self.assertEqual({},
self.test_resource._show_resource(),
'Failed to show resource')
def test_resource_handle_update(self):
mock_ppg_patch = self.test_client_plugin.update_sfc_resource
self.test_resource.resource_id = '477e8273-60a7-4c41-b683-fdb0bc7cd151'
prop_diff = {
'name': 'name-updated',
'description': 'description-updated',
}
self.test_resource.handle_update(json_snippet=None,
tmpl_diff=None,
prop_diff=prop_diff)
mock_ppg_patch.assert_called_once_with(
'port_pair_group',
{
'name': 'name-updated',
'description': 'description-updated',
}, self.test_resource.resource_id)
def test_resource_handle_update_port_pairs(self):
self.patchobject(self.test_client_plugin,
'resolve_ext_resource').return_value = ('port2')
mock_ppg_patch = self.test_client_plugin.update_sfc_resource
self.test_resource.resource_id = '477e8273-60a7-4c41-b683-fdb0bc7cd151'
prop_diff = {
port_pair_group.PortPairGroup.NAME:
'name',
port_pair_group.PortPairGroup.DESCRIPTION:
'description',
port_pair_group.PortPairGroup.PORT_PAIRS:
['port2'],
}
self.test_resource.handle_update(json_snippet=None,
tmpl_diff=None,
prop_diff=prop_diff)
mock_ppg_patch.assert_called_once_with(
'port_pair_group',
{
'name': 'name',
'description': 'description',
'port_pairs': ['port2'],
}, self.test_resource.resource_id)

View File

@ -124,6 +124,7 @@ heat.constraints =
neutron.lb.provider = heat.engine.clients.os.neutron.neutron_constraints:LBaasV1ProviderConstraint
neutron.network = heat.engine.clients.os.neutron.neutron_constraints:NetworkConstraint
neutron.port = heat.engine.clients.os.neutron.neutron_constraints:PortConstraint
neutron.port_pair = heat.engine.clients.os.neutron.neutron_constraints:PortPairConstraint
neutron.qos_policy = heat.engine.clients.os.neutron.neutron_constraints:QoSPolicyConstraint
neutron.router = heat.engine.clients.os.neutron.neutron_constraints:RouterConstraint
neutron.security_group = heat.engine.clients.os.neutron.neutron_constraints:SecurityGroupConstraint