Merge "Implementation of GBP ServiceChain resources"
This commit is contained in:
commit
95fe4101ca
|
@ -0,0 +1,166 @@
|
||||||
|
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
||||||
|
|
||||||
|
#
|
||||||
|
# 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 gbpautomation.heat.engine.resources.neutron import gbpresource
|
||||||
|
from neutronclient.common.exceptions import NeutronClientException
|
||||||
|
|
||||||
|
from heat.engine import properties
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceChainNode(gbpresource.GBPResource):
|
||||||
|
|
||||||
|
PROPERTIES = (
|
||||||
|
TENANT_ID, NAME, DESCRIPTION, SERVICE_TYPE, CONFIG
|
||||||
|
) = (
|
||||||
|
'tenant_id', 'name', 'description', 'service_type', 'config'
|
||||||
|
)
|
||||||
|
|
||||||
|
properties_schema = {
|
||||||
|
TENANT_ID: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Tenant id of the service chain node.')
|
||||||
|
),
|
||||||
|
NAME: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Name of the service chain node.'),
|
||||||
|
update_allowed=True
|
||||||
|
),
|
||||||
|
DESCRIPTION: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Description of the service chain node.'),
|
||||||
|
update_allowed=True
|
||||||
|
),
|
||||||
|
SERVICE_TYPE: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Type of service in the service chain node.'),
|
||||||
|
required=True,
|
||||||
|
update_allowed=True
|
||||||
|
),
|
||||||
|
CONFIG: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Configuration of the service chain node.'),
|
||||||
|
required=True,
|
||||||
|
update_allowed=False
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
def _show_resource(self):
|
||||||
|
client = self.grouppolicy()
|
||||||
|
sc_node_id = self.resource_id
|
||||||
|
return client.show_servicechain_node(sc_node_id)['servicechain_node']
|
||||||
|
|
||||||
|
def handle_create(self):
|
||||||
|
client = self.grouppolicy()
|
||||||
|
|
||||||
|
props = {}
|
||||||
|
for key in self.properties:
|
||||||
|
if self.properties.get(key) is not None:
|
||||||
|
props[key] = self.properties.get(key)
|
||||||
|
|
||||||
|
sc_node = client.create_servicechain_node(
|
||||||
|
{'servicechain_node': props})['servicechain_node']
|
||||||
|
|
||||||
|
self.resource_id_set(sc_node['id'])
|
||||||
|
|
||||||
|
def handle_delete(self):
|
||||||
|
|
||||||
|
client = self.grouppolicy()
|
||||||
|
sc_node_id = self.resource_id
|
||||||
|
|
||||||
|
try:
|
||||||
|
client.delete_servicechain_node(sc_node_id)
|
||||||
|
except NeutronClientException as ex:
|
||||||
|
self.client_plugin().ignore_not_found(ex)
|
||||||
|
else:
|
||||||
|
return self._delete_task()
|
||||||
|
|
||||||
|
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
|
||||||
|
if prop_diff:
|
||||||
|
self.grouppolicy().update_servicechain_node(
|
||||||
|
self.resource_id, {'servicechain_node': prop_diff})
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceChainSpec(gbpresource.GBPResource):
|
||||||
|
|
||||||
|
PROPERTIES = (
|
||||||
|
TENANT_ID, NAME, DESCRIPTION, NODES
|
||||||
|
) = (
|
||||||
|
'tenant_id', 'name', 'description', 'nodes'
|
||||||
|
)
|
||||||
|
|
||||||
|
properties_schema = {
|
||||||
|
TENANT_ID: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Tenant id of the service chain spec.')
|
||||||
|
),
|
||||||
|
NAME: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Name of the service chain spec.'),
|
||||||
|
update_allowed=True
|
||||||
|
),
|
||||||
|
DESCRIPTION: properties.Schema(
|
||||||
|
properties.Schema.STRING,
|
||||||
|
_('Description of the service chain spec.'),
|
||||||
|
update_allowed=True
|
||||||
|
),
|
||||||
|
NODES: properties.Schema(
|
||||||
|
properties.Schema.LIST,
|
||||||
|
_('Nodes in the service chain spec.'),
|
||||||
|
required=True,
|
||||||
|
update_allowed=True
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
def _show_resource(self):
|
||||||
|
client = self.grouppolicy()
|
||||||
|
sc_spec_id = self.resource_id
|
||||||
|
return client.show_servicechain_spec(sc_spec_id)['servicechain_spec']
|
||||||
|
|
||||||
|
def handle_create(self):
|
||||||
|
client = self.grouppolicy()
|
||||||
|
|
||||||
|
props = {}
|
||||||
|
for key in self.properties:
|
||||||
|
if self.properties.get(key) is not None:
|
||||||
|
props[key] = self.properties.get(key)
|
||||||
|
|
||||||
|
sc_spec = client.create_servicechain_spec(
|
||||||
|
{'servicechain_spec': props})['servicechain_spec']
|
||||||
|
|
||||||
|
self.resource_id_set(sc_spec['id'])
|
||||||
|
|
||||||
|
def handle_delete(self):
|
||||||
|
|
||||||
|
client = self.grouppolicy()
|
||||||
|
sc_spec_id = self.resource_id
|
||||||
|
|
||||||
|
try:
|
||||||
|
client.delete_servicechain_spec(sc_spec_id)
|
||||||
|
except NeutronClientException as ex:
|
||||||
|
self.client_plugin().ignore_not_found(ex)
|
||||||
|
else:
|
||||||
|
return self._delete_task()
|
||||||
|
|
||||||
|
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
|
||||||
|
if prop_diff:
|
||||||
|
self.grouppolicy().update_servicechain_spec(
|
||||||
|
self.resource_id, {'servicechain_spec': prop_diff})
|
||||||
|
|
||||||
|
|
||||||
|
def resource_mapping():
|
||||||
|
return {
|
||||||
|
'OS::Neutron::ServiceChainNode': ServiceChainNode,
|
||||||
|
'OS::Neutron::ServiceChainSpec': ServiceChainSpec,
|
||||||
|
}
|
|
@ -0,0 +1,278 @@
|
||||||
|
# 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 copy
|
||||||
|
|
||||||
|
from gbpautomation.heat.engine.resources.neutron import servicechain
|
||||||
|
from gbpclient.v2_0 import client as gbpclient
|
||||||
|
from heat.common import exception
|
||||||
|
from heat.common import template_format
|
||||||
|
from heat.tests.common import HeatTestCase
|
||||||
|
|
||||||
|
from heat.engine import scheduler
|
||||||
|
from heat.tests import utils
|
||||||
|
|
||||||
|
|
||||||
|
servicechain_node_template = '''
|
||||||
|
{
|
||||||
|
"AWSTemplateFormatVersion" : "2010-09-09",
|
||||||
|
"Description" : "Template to test neutron service chain node",
|
||||||
|
"Parameters" : {},
|
||||||
|
"Resources" : {
|
||||||
|
"servicechain_node": {
|
||||||
|
"Type": "OS::Neutron::ServiceChainNode",
|
||||||
|
"Properties": {
|
||||||
|
"name": "test-sc-node",
|
||||||
|
"description": "test service chain node resource",
|
||||||
|
"service_type": "TAP",
|
||||||
|
"config": "{'name': 'sc_node_config'}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
'''
|
||||||
|
|
||||||
|
servicechain_spec_template = '''
|
||||||
|
{
|
||||||
|
"AWSTemplateFormatVersion" : "2010-09-09",
|
||||||
|
"Description" : "Template to test neutron service chain spec",
|
||||||
|
"Parameters" : {},
|
||||||
|
"Resources" : {
|
||||||
|
"servicechain_spec": {
|
||||||
|
"Type": "OS::Neutron::ServiceChainSpec",
|
||||||
|
"Properties": {
|
||||||
|
"name": "test-sc-spec",
|
||||||
|
"description": "test service chain spec resource",
|
||||||
|
"nodes": ["1234", "7890"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
'''
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceChainNodeTest(HeatTestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(ServiceChainNodeTest, self).setUp()
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'create_servicechain_node')
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'delete_servicechain_node')
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'show_servicechain_node')
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'update_servicechain_node')
|
||||||
|
self.stub_keystoneclient()
|
||||||
|
|
||||||
|
def create_servicechain_node(self):
|
||||||
|
gbpclient.Client.create_servicechain_node({
|
||||||
|
'servicechain_node': {
|
||||||
|
"name": "test-sc-node",
|
||||||
|
"description": "test service chain node resource",
|
||||||
|
"service_type": "TAP",
|
||||||
|
"config": "{'name': 'sc_node_config'}"
|
||||||
|
}
|
||||||
|
}).AndReturn({'servicechain_node': {'id': '5678'}})
|
||||||
|
|
||||||
|
snippet = template_format.parse(servicechain_node_template)
|
||||||
|
stack = utils.parse_stack(snippet)
|
||||||
|
resource_defns = stack.t.resource_definitions(stack)
|
||||||
|
return servicechain.ServiceChainNode(
|
||||||
|
'servicechain_node', resource_defns['servicechain_node'], stack)
|
||||||
|
|
||||||
|
def test_create(self):
|
||||||
|
rsrc = self.create_servicechain_node()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
self.assertEqual((rsrc.CREATE, rsrc.COMPLETE), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_create_failed(self):
|
||||||
|
gbpclient.Client.create_servicechain_node({
|
||||||
|
'servicechain_node': {
|
||||||
|
"name": "test-sc-node",
|
||||||
|
"description": "test service chain node resource",
|
||||||
|
"service_type": "TAP",
|
||||||
|
"config": "{'name': 'sc_node_config'}"
|
||||||
|
}
|
||||||
|
}).AndRaise(servicechain.NeutronClientException())
|
||||||
|
self.m.ReplayAll()
|
||||||
|
|
||||||
|
snippet = template_format.parse(servicechain_node_template)
|
||||||
|
stack = utils.parse_stack(snippet)
|
||||||
|
resource_defns = stack.t.resource_definitions(stack)
|
||||||
|
rsrc = servicechain.ServiceChainNode(
|
||||||
|
'servicechain_node', resource_defns['servicechain_node'], stack)
|
||||||
|
|
||||||
|
error = self.assertRaises(exception.ResourceFailure,
|
||||||
|
scheduler.TaskRunner(rsrc.create))
|
||||||
|
self.assertEqual(
|
||||||
|
'NeutronClientException: An unknown exception occurred.',
|
||||||
|
str(error))
|
||||||
|
self.assertEqual((rsrc.CREATE, rsrc.FAILED), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
gbpclient.Client.delete_servicechain_node('5678')
|
||||||
|
gbpclient.Client.show_servicechain_node('5678').AndRaise(
|
||||||
|
servicechain.NeutronClientException(status_code=404))
|
||||||
|
|
||||||
|
rsrc = self.create_servicechain_node()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
scheduler.TaskRunner(rsrc.delete)()
|
||||||
|
self.assertEqual((rsrc.DELETE, rsrc.COMPLETE), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_delete_already_gone(self):
|
||||||
|
gbpclient.Client.delete_servicechain_node('5678').AndRaise(
|
||||||
|
servicechain.NeutronClientException(status_code=404))
|
||||||
|
|
||||||
|
rsrc = self.create_servicechain_node()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
scheduler.TaskRunner(rsrc.delete)()
|
||||||
|
self.assertEqual((rsrc.DELETE, rsrc.COMPLETE), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_delete_failed(self):
|
||||||
|
gbpclient.Client.delete_servicechain_node('5678').AndRaise(
|
||||||
|
servicechain.NeutronClientException(status_code=400))
|
||||||
|
|
||||||
|
rsrc = self.create_servicechain_node()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
error = self.assertRaises(exception.ResourceFailure,
|
||||||
|
scheduler.TaskRunner(rsrc.delete))
|
||||||
|
self.assertEqual(
|
||||||
|
'NeutronClientException: An unknown exception occurred.',
|
||||||
|
str(error))
|
||||||
|
self.assertEqual((rsrc.DELETE, rsrc.FAILED), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_update(self):
|
||||||
|
rsrc = self.create_servicechain_node()
|
||||||
|
gbpclient.Client.update_servicechain_node(
|
||||||
|
'5678', {'servicechain_node': {'name': 'node_update'}})
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
|
||||||
|
update_template = copy.deepcopy(rsrc.t)
|
||||||
|
update_template['Properties']['name'] = 'node_update'
|
||||||
|
scheduler.TaskRunner(rsrc.update, update_template)()
|
||||||
|
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceChainSpecTest(HeatTestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(ServiceChainSpecTest, self).setUp()
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'create_servicechain_spec')
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'delete_servicechain_spec')
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'show_servicechain_spec')
|
||||||
|
self.m.StubOutWithMock(gbpclient.Client, 'update_servicechain_spec')
|
||||||
|
self.stub_keystoneclient()
|
||||||
|
|
||||||
|
def create_servicechain_spec(self):
|
||||||
|
gbpclient.Client.create_servicechain_spec({
|
||||||
|
"servicechain_spec": {
|
||||||
|
"name": "test-sc-spec",
|
||||||
|
"description": "test service chain spec resource",
|
||||||
|
"nodes": ["1234", "7890"]
|
||||||
|
}
|
||||||
|
}).AndReturn({'servicechain_spec': {'id': '5678'}})
|
||||||
|
|
||||||
|
snippet = template_format.parse(servicechain_spec_template)
|
||||||
|
stack = utils.parse_stack(snippet)
|
||||||
|
resource_defns = stack.t.resource_definitions(stack)
|
||||||
|
return servicechain.ServiceChainSpec(
|
||||||
|
'servicechain_spec', resource_defns['servicechain_spec'], stack)
|
||||||
|
|
||||||
|
def test_create(self):
|
||||||
|
rsrc = self.create_servicechain_spec()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
self.assertEqual((rsrc.CREATE, rsrc.COMPLETE), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_create_failed(self):
|
||||||
|
gbpclient.Client.create_servicechain_spec({
|
||||||
|
'servicechain_spec': {
|
||||||
|
"name": "test-sc-spec",
|
||||||
|
"description": "test service chain spec resource",
|
||||||
|
"nodes": ["1234", "7890"]
|
||||||
|
}
|
||||||
|
}).AndRaise(servicechain.NeutronClientException())
|
||||||
|
self.m.ReplayAll()
|
||||||
|
|
||||||
|
snippet = template_format.parse(servicechain_spec_template)
|
||||||
|
stack = utils.parse_stack(snippet)
|
||||||
|
resource_defns = stack.t.resource_definitions(stack)
|
||||||
|
rsrc = servicechain.ServiceChainSpec(
|
||||||
|
'servicechain_spec', resource_defns['servicechain_spec'], stack)
|
||||||
|
|
||||||
|
error = self.assertRaises(exception.ResourceFailure,
|
||||||
|
scheduler.TaskRunner(rsrc.create))
|
||||||
|
self.assertEqual(
|
||||||
|
'NeutronClientException: An unknown exception occurred.',
|
||||||
|
str(error))
|
||||||
|
self.assertEqual((rsrc.CREATE, rsrc.FAILED), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
gbpclient.Client.delete_servicechain_spec('5678')
|
||||||
|
gbpclient.Client.show_servicechain_spec('5678').AndRaise(
|
||||||
|
servicechain.NeutronClientException(status_code=404))
|
||||||
|
|
||||||
|
rsrc = self.create_servicechain_spec()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
scheduler.TaskRunner(rsrc.delete)()
|
||||||
|
self.assertEqual((rsrc.DELETE, rsrc.COMPLETE), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_delete_already_gone(self):
|
||||||
|
gbpclient.Client.delete_servicechain_spec('5678').AndRaise(
|
||||||
|
servicechain.NeutronClientException(status_code=404))
|
||||||
|
|
||||||
|
rsrc = self.create_servicechain_spec()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
scheduler.TaskRunner(rsrc.delete)()
|
||||||
|
self.assertEqual((rsrc.DELETE, rsrc.COMPLETE), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_delete_failed(self):
|
||||||
|
gbpclient.Client.delete_servicechain_spec('5678').AndRaise(
|
||||||
|
servicechain.NeutronClientException(status_code=400))
|
||||||
|
|
||||||
|
rsrc = self.create_servicechain_spec()
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
error = self.assertRaises(exception.ResourceFailure,
|
||||||
|
scheduler.TaskRunner(rsrc.delete))
|
||||||
|
self.assertEqual(
|
||||||
|
'NeutronClientException: An unknown exception occurred.',
|
||||||
|
str(error))
|
||||||
|
self.assertEqual((rsrc.DELETE, rsrc.FAILED), rsrc.state)
|
||||||
|
self.m.VerifyAll()
|
||||||
|
|
||||||
|
def test_update(self):
|
||||||
|
rsrc = self.create_servicechain_spec()
|
||||||
|
gbpclient.Client.update_servicechain_spec(
|
||||||
|
'5678', {'servicechain_spec': {'name': 'spec_update'}})
|
||||||
|
self.m.ReplayAll()
|
||||||
|
scheduler.TaskRunner(rsrc.create)()
|
||||||
|
|
||||||
|
update_template = copy.deepcopy(rsrc.t)
|
||||||
|
update_template['Properties']['name'] = 'spec_update'
|
||||||
|
scheduler.TaskRunner(rsrc.update, update_template)()
|
||||||
|
|
||||||
|
self.m.VerifyAll()
|
Loading…
Reference in New Issue