diff options
-rw-r--r-- | snaps/domain/test/volume_tests.py | 63 | ||||
-rw-r--r-- | snaps/domain/volume.py | 59 | ||||
-rw-r--r-- | snaps/openstack/create_volume_type.py | 235 | ||||
-rw-r--r-- | snaps/openstack/tests/create_volume_type_tests.py | 323 | ||||
-rw-r--r-- | snaps/openstack/utils/cinder_utils.py | 200 | ||||
-rw-r--r-- | snaps/openstack/utils/tests/cinder_utils_tests.py | 320 | ||||
-rw-r--r-- | snaps/test_suite_builder.py | 44 |
7 files changed, 1209 insertions, 35 deletions
diff --git a/snaps/domain/test/volume_tests.py b/snaps/domain/test/volume_tests.py index f105e38..ec5f7b7 100644 --- a/snaps/domain/test/volume_tests.py +++ b/snaps/domain/test/volume_tests.py @@ -14,7 +14,68 @@ # limitations under the License. import unittest -from snaps.domain.volume import QoSSpec +from snaps.domain.volume import QoSSpec, VolumeType, VolumeTypeEncryption + + +class VolumeTypeDomainObjectTests(unittest.TestCase): + """ + Tests the construction of the snaps.domain.volume.VolumeType class + """ + + def test_construction_positional(self): + encryption = VolumeTypeEncryption( + 'id-encrypt1', 'id-vol-type1', 'loc1', 'provider1', 'cipher1', 99) + qos_spec = QoSSpec('name', 'id', 'consumer') + + volume_type = VolumeType('name', 'id', True, encryption, qos_spec) + self.assertEqual('name', volume_type.name) + self.assertEqual('id', volume_type.id) + self.assertTrue(volume_type.public) + self.assertEqual(encryption, volume_type.encryption) + self.assertEqual(qos_spec, volume_type.qos_spec) + + def test_construction_named(self): + encryption = VolumeTypeEncryption( + 'id-encrypt1', 'id-vol-type1', 'loc1', 'provider1', 'cipher1', 99) + qos_spec = QoSSpec('name', 'id', 'consumer') + + volume_type = VolumeType( + qos_spec=qos_spec, encryption=encryption, volume_type_id='id', + name='name', public='true') + self.assertEqual('name', volume_type.name) + self.assertEqual('id', volume_type.id) + self.assertTrue(volume_type.public) + self.assertEqual(encryption, volume_type.encryption) + self.assertEqual(qos_spec, volume_type.qos_spec) + + +class VolumeTypeEncryptionObjectTests(unittest.TestCase): + """ + Tests the construction of the snaps.domain.volume.VolumeTypeEncryption + class + """ + + def test_construction_positional(self): + encryption = VolumeTypeEncryption( + 'id-encrypt1', 'id-vol-type1', 'loc1', 'provider1', 'cipher1', 99) + self.assertEqual('id-encrypt1', encryption.id) + self.assertEqual('id-vol-type1', encryption.volume_type_id) + self.assertEqual('loc1', encryption.control_location) + self.assertEqual('provider1', encryption.provider) + self.assertEqual('cipher1', encryption.cipher) + self.assertEqual(99, encryption.key_size) + + def test_construction_named(self): + encryption = VolumeTypeEncryption( + key_size=89, cipher='cipher2', provider='provider2', + control_location='loc2', volume_type_id='id-vol-type2', + volume_encryption_id='id-encrypt2') + self.assertEqual('id-encrypt2', encryption.id) + self.assertEqual('id-vol-type2', encryption.volume_type_id) + self.assertEqual('loc2', encryption.control_location) + self.assertEqual('provider2', encryption.provider) + self.assertEqual('cipher2', encryption.cipher) + self.assertEqual(89, encryption.key_size) class QoSSpecDomainObjectTests(unittest.TestCase): diff --git a/snaps/domain/volume.py b/snaps/domain/volume.py index 9b35c9b..e82a60a 100644 --- a/snaps/domain/volume.py +++ b/snaps/domain/volume.py @@ -14,6 +14,65 @@ # limitations under the License. +class VolumeType: + """ + SNAPS domain object for Volume Types. Should contain attributes that + are shared amongst cloud providers + """ + def __init__(self, name, volume_type_id, public, encryption, qos_spec): + """ + Constructor + :param name: the volume's name + :param volume_type_id: the volume type's id + :param public: True if public + :param encryption: instance of a VolumeTypeEncryption domain object + :param qos_spec: instance of a QoSSpec domain object + """ + self.name = name + self.id = volume_type_id + self.public = public + self.encryption = encryption + self.qos_spec = qos_spec + + def __eq__(self, other): + return (self.name == other.name and self.id == other.id + and self.public == other.public + and self.encryption == other.encryption + and self.qos_spec == other.qos_spec) + + +class VolumeTypeEncryption: + """ + SNAPS domain object for Volume Types. Should contain attributes that + are shared amongst cloud providers + """ + def __init__(self, volume_encryption_id, volume_type_id, + control_location, provider, cipher, key_size): + """ + Constructor + :param volume_encryption_id: the encryption id + :param volume_type_id: the associated volume type's id + :param control_location: front-end | back-end + :param provider: the encryption provider class + :param cipher: the encryption cipher + :param key_size: the encryption key size + """ + self.id = volume_encryption_id + self.volume_type_id = volume_type_id + self.control_location = control_location + self.provider = provider + self.cipher = cipher + self.key_size = key_size + + def __eq__(self, other): + return (self.id == other.id + and self.volume_type_id == other.volume_type_id + and self.control_location == other.control_location + and self.provider == other.provider + and self.cipher == other.cipher + and self.key_size == other.key_size) + + class QoSSpec: """ SNAPS domain object for Volume Types. Should contain attributes that diff --git a/snaps/openstack/create_volume_type.py b/snaps/openstack/create_volume_type.py new file mode 100644 index 0000000..a60bb1e --- /dev/null +++ b/snaps/openstack/create_volume_type.py @@ -0,0 +1,235 @@ +# Copyright (c) 2017 Cable Television Laboratories, Inc. ("CableLabs") +# and others. 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 logging + +import enum +from cinderclient.exceptions import NotFound +from neutronclient.common.utils import str2bool + +from snaps.openstack.openstack_creator import OpenStackVolumeObject +from snaps.openstack.utils import cinder_utils + +__author__ = 'spisarski' + +logger = logging.getLogger('create_volume_type') + + +class OpenStackVolumeType(OpenStackVolumeObject): + """ + Class responsible for managing an volume in OpenStack + """ + + def __init__(self, os_creds, volume_type_settings): + """ + Constructor + :param os_creds: The OpenStack connection credentials + :param volume_type_settings: The volume type settings + :return: + """ + super(self.__class__, self).__init__(os_creds) + + self.volume_type_settings = volume_type_settings + self.__volume_type = None + + def initialize(self): + """ + Loads the existing Volume + :return: The Volume domain object or None + """ + super(self.__class__, self).initialize() + + self.__volume_type = cinder_utils.get_volume_type( + self._cinder, volume_type_settings=self.volume_type_settings) + + return self.__volume_type + + def create(self, block=False): + """ + Creates the volume in OpenStack if it does not already exist and + returns the domain Volume object + :return: The Volume domain object or None + """ + self.initialize() + + if not self.__volume_type: + self.__volume_type = cinder_utils.create_volume_type( + self._cinder, self.volume_type_settings) + logger.info( + 'Created volume type with name - %s', + self.volume_type_settings.name) + + return self.__volume_type + + def clean(self): + """ + Cleanse environment of all artifacts + :return: void + """ + if self.__volume_type: + try: + cinder_utils.delete_volume_type(self._cinder, + self.__volume_type) + except NotFound: + pass + + self.__volume_type = None + + def get_volume_type(self): + """ + Returns the domain Volume object as it was populated when create() was + called + :return: the object + """ + return self.__volume_type + + +class VolumeTypeSettings: + def __init__(self, **kwargs): + """ + Constructor + :param name: the volume's name (required) + :param description: the volume's name (optional) + :param encryption: VolumeTypeEncryptionSettings (optional) + :param qos_spec_name: name of the QoS Spec to associate (optional) + :param public: When True, an image will be created with public + visibility (default - False) + + TODO - Implement project_access parameter that will associate this + VolumeType to a list of project names + """ + + self.name = kwargs.get('name') + self.description = kwargs.get('description') + self.qos_spec_name = kwargs.get('qos_spec_name') + + if 'encryption' in kwargs: + if isinstance(kwargs['encryption'], dict): + self.encryption = VolumeTypeEncryptionSettings( + **kwargs['encryption']) + elif isinstance(kwargs['encryption'], + VolumeTypeEncryptionSettings): + self.encryption = kwargs['encryption'] + else: + self.encryption = None + + if 'public' in kwargs: + if isinstance(kwargs['public'], str): + self.public = str2bool(kwargs['public']) + else: + self.public = kwargs['public'] + else: + self.public = False + + if not self.name: + raise VolumeTypeSettingsError("The attribute name is required") + + def __eq__(self, other): + return (self.name == other.name + and self.description == other.description + and self.qos_spec_name == other.qos_spec_name + and self.encryption == other.encryption + and self.public == other.public) + + +class ControlLocation(enum.Enum): + """ + QoS Specification consumer types + """ + front_end = 'front-end' + back_end = 'back-end' + + +class VolumeTypeEncryptionSettings: + def __init__(self, **kwargs): + """ + Constructor + :param name: the volume's name (required) + :param provider_class: the volume's provider class (e.g. LuksEncryptor) + :param control_location: the notional service where encryption is + performed (e.g., front-end=Nova). The default + value is 'front-end.' + :param cipher: the encryption algorithm/mode to use + (e.g., aes-xts-plain64). If the field is left empty, + the provider default will be used + :param key_size: the size of the encryption key, in bits + (e.g., 128, 256). If the field is left empty, the + provider default will be used + """ + + self.name = kwargs.get('name') + self.provider_class = kwargs.get('provider_class') + self.control_location = kwargs.get('control_location') + if kwargs.get('control_location'): + self.control_location = map_control_location( + kwargs['control_location']) + else: + self.control_location = None + + self.cipher = kwargs.get('cipher') + self.key_size = kwargs.get('key_size') + + if (not self.name or not self.provider_class + or not self.control_location): + raise VolumeTypeSettingsError( + 'The attributes name, provider_class, and control_location ' + 'are required') + + def __eq__(self, other): + return (self.name == other.name + and self.provider_class == other.provider_class + and self.control_location == other.control_location + and self.cipher == other.cipher + and self.key_size == other.key_size) + + +def map_control_location(control_location): + """ + Takes a the protocol value maps it to the Consumer enum. When None return + None + :param control_location: the value to map to the Enum + :return: a ControlLocation enum object + :raise: Exception if control_location parameter is invalid + """ + if not control_location: + return None + elif isinstance(control_location, ControlLocation): + return control_location + else: + proto_str = str(control_location) + if proto_str == 'front-end': + return ControlLocation.front_end + elif proto_str == 'back-end': + return ControlLocation.back_end + else: + raise VolumeTypeSettingsError('Invalid Consumer - ' + proto_str) + + +class VolumeTypeSettingsError(Exception): + """ + Exception to be thrown when an volume settings are incorrect + """ + + def __init__(self, message): + Exception.__init__(self, message) + + +class VolumeTypeCreationError(Exception): + """ + Exception to be thrown when an volume cannot be created + """ + + def __init__(self, message): + Exception.__init__(self, message) diff --git a/snaps/openstack/tests/create_volume_type_tests.py b/snaps/openstack/tests/create_volume_type_tests.py new file mode 100644 index 0000000..93e9351 --- /dev/null +++ b/snaps/openstack/tests/create_volume_type_tests.py @@ -0,0 +1,323 @@ +# Copyright (c) 2017 Cable Television Laboratories, Inc. ("CableLabs") +# and others. 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. +from snaps.openstack.create_qos import QoSSettings, Consumer, OpenStackQoS + +try: + from urllib.request import URLError +except ImportError: + from urllib2 import URLError + +import logging +import unittest +import uuid + +from snaps.openstack import create_volume_type +from snaps.openstack.create_volume_type import ( + VolumeTypeSettings, VolumeTypeSettingsError, VolumeTypeEncryptionSettings, + ControlLocation) +from snaps.openstack.tests.os_source_file_test import OSIntegrationTestCase +from snaps.openstack.utils import cinder_utils + +__author__ = 'spisarski' + +logger = logging.getLogger('create_volume_type_tests') + + +class VolumeTypeSettingsUnitTests(unittest.TestCase): + """ + Tests the construction of the VolumeTypeSettings class + """ + + def test_no_params(self): + with self.assertRaises(VolumeTypeSettingsError): + VolumeTypeSettings() + + def test_empty_config(self): + with self.assertRaises(VolumeTypeSettingsError): + VolumeTypeSettings(**dict()) + + def test_name_only(self): + settings = VolumeTypeSettings(name='foo') + self.assertEqual('foo', settings.name) + self.assertIsNone(settings.description) + self.assertIsNone(settings.qos_spec_name) + self.assertIsNone(settings.encryption) + self.assertFalse(settings.public) + + def test_config_with_name_only(self): + settings = VolumeTypeSettings(**{'name': 'foo'}) + self.assertEqual('foo', settings.name) + self.assertIsNone(settings.description) + self.assertIsNone(settings.qos_spec_name) + self.assertIsNone(settings.encryption) + self.assertFalse(settings.public) + + def test_all(self): + encryption_settings = VolumeTypeEncryptionSettings( + name='foo', provider_class='bar', + control_location=ControlLocation.back_end) + settings = VolumeTypeSettings( + name='foo', description='desc', encryption=encryption_settings, + qos_spec_name='spec_name', public=True) + self.assertEqual('foo', settings.name) + self.assertEqual('desc', settings.description) + self.assertEqual('spec_name', settings.qos_spec_name) + self.assertEqual(encryption_settings, settings.encryption) + self.assertTrue(True, settings.public) + + def test_all_string(self): + encryption_settings = { + 'name': 'foo', 'provider_class': 'bar', + 'control_location': 'back-end'} + settings = VolumeTypeSettings( + name='foo', description='desc', encryption=encryption_settings, + qos_spec_name='spec_name', public='true') + self.assertEqual('foo', settings.name) + self.assertEqual('desc', settings.description) + self.assertEqual('spec_name', settings.qos_spec_name) + self.assertEqual(VolumeTypeEncryptionSettings(**encryption_settings), + settings.encryption) + self.assertTrue(settings.public) + + def test_config_all(self): + encryption_settings = { + 'name': 'foo', 'provider_class': 'bar', + 'control_location': 'back-end'} + settings = VolumeTypeSettings( + **{'name': 'foo', 'description': 'desc', + 'encryption': encryption_settings, + 'qos_spec_name': 'spec_name', 'public': 'false'}) + self.assertEqual('foo', settings.name) + self.assertEqual('desc', settings.description) + self.assertEqual('spec_name', settings.qos_spec_name) + self.assertEqual(VolumeTypeEncryptionSettings(**encryption_settings), + settings.encryption) + self.assertFalse(settings.public) + + +class CreateSimpleVolumeTypeSuccessTests(OSIntegrationTestCase): + """ + Test for the OpenStackVolumeType class defined in create_volume_type.py + without any QoS Specs or Encryption + """ + + def setUp(self): + """ + Instantiates the CreateVolumeType object that is responsible for + downloading and creating an OS volume type file within OpenStack + """ + super(self.__class__, self).__start__() + + guid = uuid.uuid4() + self.volume_type_settings = VolumeTypeSettings( + name=self.__class__.__name__ + '-' + str(guid)) + + self.cinder = cinder_utils.cinder_client(self.os_creds) + self.volume_type_creator = None + + def tearDown(self): + """ + Cleans the volume type + """ + if self.volume_type_creator: + self.volume_type_creator.clean() + + super(self.__class__, self).__clean__() + + def test_create_volume_type(self): + """ + Tests the creation of an OpenStack volume. + """ + # Create VolumeType + self.volume_type_creator = create_volume_type.OpenStackVolumeType( + self.os_creds, self.volume_type_settings) + created_volume_type = self.volume_type_creator.create() + self.assertIsNotNone(created_volume_type) + self.assertEqual(self.volume_type_settings.name, + created_volume_type.name) + + retrieved_volume_type1 = cinder_utils.get_volume_type( + self.cinder, volume_type_settings=self.volume_type_settings) + self.assertIsNotNone(retrieved_volume_type1) + self.assertEqual(created_volume_type, retrieved_volume_type1) + + retrieved_volume_type2 = cinder_utils.get_volume_type_by_id( + self.cinder, created_volume_type.id) + self.assertEqual(created_volume_type, retrieved_volume_type2) + + def test_create_delete_volume_type(self): + """ + Tests the creation then deletion of an OpenStack volume type to ensure + clean() does not raise an Exception. + """ + # Create VolumeType + self.volume_type_creator = create_volume_type.OpenStackVolumeType( + self.os_creds, self.volume_type_settings) + created_volume_type = self.volume_type_creator.create() + self.assertIsNotNone(created_volume_type) + + retrieved_volume_type = cinder_utils.get_volume_type( + self.cinder, volume_type_settings=self.volume_type_settings) + self.assertIsNotNone(retrieved_volume_type) + self.assertEqual(created_volume_type, retrieved_volume_type) + + # Delete VolumeType manually + cinder_utils.delete_volume_type(self.cinder, created_volume_type) + + self.assertIsNone(cinder_utils.get_volume_type( + self.cinder, volume_type_settings=self.volume_type_settings)) + + # Must not throw an exception when attempting to cleanup non-existent + # volume_type + self.volume_type_creator.clean() + self.assertIsNone(self.volume_type_creator.get_volume_type()) + + def test_create_same_volume_type(self): + """ + Tests the creation of an OpenStack volume_type when one already exists. + """ + # Create VolumeType + self.volume_type_creator = create_volume_type.OpenStackVolumeType( + self.os_creds, self.volume_type_settings) + volume_type1 = self.volume_type_creator.create() + + retrieved_volume_type = cinder_utils.get_volume_type( + self.cinder, volume_type_settings=self.volume_type_settings) + self.assertEqual(volume_type1, retrieved_volume_type) + + # Should be retrieving the instance data + os_volume_type_2 = create_volume_type.OpenStackVolumeType( + self.os_creds, self.volume_type_settings) + volume_type2 = os_volume_type_2.create() + self.assertEqual(volume_type2, volume_type2) + + +class CreateVolumeTypeComplexTests(OSIntegrationTestCase): + """ + Test cases for the CreateVolumeType class that include QoS Specs and/or + encryption + """ + + def setUp(self): + super(self.__class__, self).__start__() + + self.cinder = cinder_utils.cinder_client(self.os_creds) + + guid = self.__class__.__name__ + '-' + str(uuid.uuid4()) + + self.volume_type_name = guid + '-vol_type' + self.volume_type_creator = None + + qos_settings = QoSSettings( + name=guid + '-qos-spec', consumer=Consumer.both) + self.qos_creator = OpenStackQoS(self.os_creds, qos_settings) + self.qos_creator.create() + + def tearDown(self): + if self.volume_type_creator: + self.volume_type_creator.clean() + + if self.qos_creator: + self.qos_creator.clean() + + super(self.__class__, self).__clean__() + + def test_volume_type_with_qos(self): + """ + Creates a Volume Type object with an associated QoS Spec + """ + self.volume_type_creator = create_volume_type.OpenStackVolumeType( + self.os_creds, + VolumeTypeSettings( + name=self.volume_type_name, + qos_spec_name=self.qos_creator.qos_settings.name)) + + vol_type = self.volume_type_creator.create() + self.assertEqual(self.volume_type_creator.volume_type_settings.name, + vol_type.name) + self.assertEqual(self.volume_type_creator.volume_type_settings.name, + vol_type.name) + self.assertIsNotNone(vol_type.qos_spec) + self.assertEqual( + self.volume_type_creator.volume_type_settings.qos_spec_name, + vol_type.qos_spec.name) + self.assertIsNone(vol_type.encryption) + + vol_type_query = cinder_utils.get_volume_type_by_id( + self.cinder, vol_type.id) + self.assertIsNotNone(vol_type_query) + self.assertEqual(vol_type, vol_type_query) + + def test_volume_type_with_encryption(self): + """ + Creates a Volume Type object with encryption + """ + encryption_settings = VolumeTypeEncryptionSettings( + name='foo', provider_class='bar', + control_location=ControlLocation.back_end) + self.volume_type_creator = create_volume_type.OpenStackVolumeType( + self.os_creds, + VolumeTypeSettings( + name=self.volume_type_name, + encryption=encryption_settings)) + + vol_type = self.volume_type_creator.create() + self.assertEqual(self.volume_type_creator.volume_type_settings.name, + vol_type.name) + self.assertEqual(self.volume_type_creator.volume_type_settings.name, + vol_type.name) + self.assertIsNone(vol_type.qos_spec) + self.assertIsNotNone(vol_type.encryption) + + self.assertEqual(encryption_settings.control_location.value, + vol_type.encryption.control_location) + + vol_type_query = cinder_utils.get_volume_type_by_id( + self.cinder, vol_type.id) + self.assertIsNotNone(vol_type_query) + self.assertEqual(vol_type, vol_type_query) + + def test_volume_type_with_qos_and_encryption(self): + """ + Creates a Volume Type object with encryption and an associated QoS Spec + """ + encryption_settings = VolumeTypeEncryptionSettings( + name='foo', provider_class='bar', + control_location=ControlLocation.back_end) + self.volume_type_creator = create_volume_type.OpenStackVolumeType( + self.os_creds, + VolumeTypeSettings( + name=self.volume_type_name, + encryption=encryption_settings, + qos_spec_name=self.qos_creator.qos_settings.name)) + + vol_type = self.volume_type_creator.create() + self.assertEqual(self.volume_type_creator.volume_type_settings.name, + vol_type.name) + self.assertEqual(self.volume_type_creator.volume_type_settings.name, + vol_type.name) + self.assertIsNotNone(vol_type.qos_spec) + self.assertEqual( + self.volume_type_creator.volume_type_settings.qos_spec_name, + vol_type.qos_spec.name) + self.assertIsNotNone(vol_type.encryption) + + self.assertEqual(encryption_settings.control_location.value, + vol_type.encryption.control_location) + + vol_type_query = cinder_utils.get_volume_type_by_id( + self.cinder, vol_type.id) + self.assertIsNotNone(vol_type_query) + self.assertEqual(vol_type, vol_type_query) diff --git a/snaps/openstack/utils/cinder_utils.py b/snaps/openstack/utils/cinder_utils.py index 5f847a1..d13277d 100644 --- a/snaps/openstack/utils/cinder_utils.py +++ b/snaps/openstack/utils/cinder_utils.py @@ -15,15 +15,15 @@ import logging from cinderclient.client import Client +from cinderclient.exceptions import NotFound -from snaps.domain.volume import QoSSpec +from snaps.domain.volume import QoSSpec, VolumeType, VolumeTypeEncryption from snaps.openstack.utils import keystone_utils __author__ = 'spisarski' logger = logging.getLogger('cinder_utils') -VERSION_1 = 1 VERSION_2 = 2 VERSION_3 = 3 @@ -42,7 +42,172 @@ def cinder_client(os_creds): region_name=os_creds.region_name) -def get_qos(cinder, qos_name=None, qos_settings=None): +def get_volume_type(cinder, volume_type_name=None, volume_type_settings=None): + """ + Returns an OpenStack volume type object for a given name + :param cinder: the Cinder client + :param volume_type_name: the volume type name to lookup + :param volume_type_settings: the volume type settings used for lookups + :return: the volume type object or None + """ + if not volume_type_name and not volume_type_settings: + return None + + if volume_type_settings: + volume_type_name = volume_type_settings.name + + volume_types = cinder.volume_types.list() + for vol_type in volume_types: + if vol_type.name == volume_type_name: + encryption = get_volume_encryption_by_type(cinder, vol_type) + return VolumeType(vol_type.name, vol_type.id, vol_type.is_public, + encryption, None) + + +def __get_os_volume_type_by_id(cinder, volume_type_id): + """ + Returns an OpenStack volume type object for a given name + :param cinder: the Cinder client + :param volume_type_id: the volume_type ID to lookup + :return: the SNAPS-OO Domain Volume object or None + """ + try: + return cinder.volume_types.get(volume_type_id) + except NotFound: + logger.info('Volume with ID [%s] does not exist', + volume_type_id) + + +def get_volume_type_by_id(cinder, volume_type_id): + """ + Returns an OpenStack volume type object for a given name + :param cinder: the Cinder client + :param volume_type_id: the volume_type ID to lookup + :return: the SNAPS-OO Domain Volume object or None + """ + os_vol_type = __get_os_volume_type_by_id(cinder, volume_type_id) + if os_vol_type: + temp_vol_type = VolumeType(os_vol_type.name, os_vol_type.id, + os_vol_type.is_public, None, None) + encryption = get_volume_encryption_by_type(cinder, temp_vol_type) + + qos_spec = None + if os_vol_type.qos_specs_id: + qos_spec = get_qos_by_id(cinder, os_vol_type.qos_specs_id) + + return VolumeType(os_vol_type.name, os_vol_type.id, + os_vol_type.is_public, encryption, qos_spec) + + +def create_volume_type(cinder, type_settings): + """ + Creates and returns OpenStack volume type object with an external URL + :param cinder: the cinder client + :param type_settings: the volume type settings object + :return: the volume type domain object + :raise Exception if using a file and it cannot be found + """ + vol_type = cinder.volume_types.create( + type_settings.name, type_settings.description, + type_settings.public) + + vol_encryption = None + if type_settings.encryption: + try: + vol_encryption = create_volume_encryption( + cinder, vol_type, type_settings.encryption) + except Exception as e: + logger.warn('Error creating volume encryption - %s', e) + + qos_spec = None + if type_settings.qos_spec_name: + try: + qos_spec = get_qos(cinder, qos_name=type_settings.qos_spec_name) + cinder.qos_specs.associate(qos_spec, vol_type.id) + except NotFound as e: + logger.warn('Unable to locate qos_spec named %s - %s', + type_settings.qos_spec_name, e) + + return VolumeType(vol_type.name, vol_type.id, vol_type.is_public, + vol_encryption, qos_spec) + + +def delete_volume_type(cinder, vol_type): + """ + Deletes an volume from OpenStack + :param cinder: the cinder client + :param vol_type: the VolumeType domain object + """ + logger.info('Deleting volume named - %s', vol_type.name) + cinder.volume_types.delete(vol_type.id) + + +def get_volume_encryption_by_type(cinder, volume_type): + """ + Returns an OpenStack volume type object for a given name + :param cinder: the Cinder client + :param volume_type: the VolumeType domain object + :return: the VolumeEncryption domain object or None + """ + os_vol_type = __get_os_volume_type_by_id(cinder, volume_type.id) + encryption = cinder.volume_encryption_types.get(os_vol_type) + if hasattr(encryption, 'encryption_id'): + cipher = None + if hasattr(encryption, 'cipher'): + cipher = encryption.cipher + key_size = None + if hasattr(encryption, 'key_size'): + key_size = encryption.key_size + return VolumeTypeEncryption( + encryption.encryption_id, encryption.volume_type_id, + encryption.control_location, encryption.provider, cipher, key_size) + + +def create_volume_encryption(cinder, volume_type, encryption_settings): + """ + Creates and returns OpenStack volume type object with an external URL + :param cinder: the cinder client + :param volume_type: the VolumeType object to associate the encryption + :param encryption_settings: the volume type encryption settings object + :return: the VolumeTypeEncryption domain object + """ + specs = {'name': encryption_settings.name, + 'provider': encryption_settings.provider_class} + if encryption_settings.key_size: + specs['key_size'] = encryption_settings.key_size + if encryption_settings.provider_class: + specs['provider_class'] = encryption_settings.provider_class + if encryption_settings.control_location: + specs['control_location'] = encryption_settings.control_location.value + if encryption_settings.cipher: + specs['cipher'] = encryption_settings.cipher + + encryption = cinder.volume_encryption_types.create(volume_type.id, specs) + + cipher = None + if hasattr(encryption, 'cipher'): + cipher = encryption.cipher + key_size = None + if hasattr(encryption, 'key_size'): + key_size = encryption.key_size + return VolumeTypeEncryption( + encryption.encryption_id, encryption.volume_type_id, + encryption.control_location, encryption.provider, cipher, key_size) + + +def delete_volume_type_encryption(cinder, vol_type): + """ + Deletes an volume from OpenStack + :param cinder: the cinder client + :param vol_type: the associated VolumeType domain object + """ + logger.info('Deleting volume encryption for volume type - %s', + vol_type.name) + os_vol_type = __get_os_volume_type_by_id(cinder, vol_type.id) + cinder.volume_encryption_types.delete(os_vol_type) + + +def __get_os_qos(cinder, qos_name=None, qos_settings=None): """ Returns an OpenStack QoS object for a given name :param cinder: the Cinder client @@ -53,20 +218,27 @@ def get_qos(cinder, qos_name=None, qos_settings=None): if not qos_name and not qos_settings: return None - qos_name = qos_name if qos_settings: qos_name = qos_settings.name qoss = cinder.qos_specs.list() for qos in qoss: if qos.name == qos_name: - if qos_settings: - if qos_settings.consumer.value == qos.consumer: - return QoSSpec(name=qos.name, spec_id=qos.id, - consumer=qos.consumer) - else: - return QoSSpec(name=qos.name, spec_id=qos.id, - consumer=qos.consumer) + return qos + + +def get_qos(cinder, qos_name=None, qos_settings=None): + """ + Returns an OpenStack QoS object for a given name + :param cinder: the Cinder client + :param qos_name: the qos name to lookup + :param qos_settings: the qos settings used for lookups + :return: the qos object or None + """ + os_qos = __get_os_qos(cinder, qos_name, qos_settings) + if os_qos: + return QoSSpec(name=os_qos.name, spec_id=os_qos.id, + consumer=os_qos.consumer) def get_qos_by_id(cinder, qos_id): @@ -102,9 +274,3 @@ def delete_qos(cinder, qos): """ logger.info('Deleting QoS named - %s', qos.name) cinder.qos_specs.delete(qos.id) - - -class CinderException(Exception): - """ - Exception when calls to the Cinder client cannot be served properly - """ diff --git a/snaps/openstack/utils/tests/cinder_utils_tests.py b/snaps/openstack/utils/tests/cinder_utils_tests.py index e6ad2a0..a45167e 100644 --- a/snaps/openstack/utils/tests/cinder_utils_tests.py +++ b/snaps/openstack/utils/tests/cinder_utils_tests.py @@ -15,9 +15,11 @@ import logging import uuid -from cinderclient.exceptions import NotFound +from cinderclient.exceptions import NotFound, BadRequest from snaps.openstack.create_qos import QoSSettings, Consumer +from snaps.openstack.create_volume_type import ( + VolumeTypeSettings, VolumeTypeEncryptionSettings, ControlLocation) from snaps.openstack.tests import validation_utils from snaps.openstack.tests.os_source_file_test import OSComponentTestCase from snaps.openstack.utils import cinder_utils @@ -86,15 +88,14 @@ class CinderUtilsQoSTests(OSComponentTestCase): """ qos_settings = QoSSettings(name=self.qos_name, specs=self.specs, consumer=Consumer.both) - self.qos = cinder_utils.create_qos( - self.cinder, qos_settings) + self.qos = cinder_utils.create_qos(self.cinder, qos_settings) self.assertIsNotNone(self.qos) qos1 = cinder_utils.get_qos(self.cinder, qos_settings=qos_settings) self.assertIsNotNone(qos1) validation_utils.objects_equivalent(self.qos, qos1) - qos2 = cinder_utils.get_qos(self.cinder, qos_name=qos_settings.name) + qos2 = cinder_utils.get_qos_by_id(self.cinder, qos1.id) self.assertIsNotNone(qos2) validation_utils.objects_equivalent(self.qos, qos2) @@ -104,15 +105,14 @@ class CinderUtilsQoSTests(OSComponentTestCase): """ qos_settings = QoSSettings(name=self.qos_name, specs=self.specs, consumer=Consumer.front_end) - self.qos = cinder_utils.create_qos( - self.cinder, qos_settings) + self.qos = cinder_utils.create_qos(self.cinder, qos_settings) self.assertIsNotNone(self.qos) qos1 = cinder_utils.get_qos(self.cinder, qos_settings=qos_settings) self.assertIsNotNone(qos1) validation_utils.objects_equivalent(self.qos, qos1) - qos2 = cinder_utils.get_qos(self.cinder, qos_name=qos_settings.name) + qos2 = cinder_utils.get_qos_by_id(self.cinder, qos1.id) self.assertIsNotNone(qos2) validation_utils.objects_equivalent(self.qos, qos2) @@ -122,15 +122,14 @@ class CinderUtilsQoSTests(OSComponentTestCase): """ qos_settings = QoSSettings(name=self.qos_name, specs=self.specs, consumer=Consumer.back_end) - self.qos = cinder_utils.create_qos( - self.cinder, qos_settings) + self.qos = cinder_utils.create_qos(self.cinder, qos_settings) self.assertIsNotNone(self.qos) qos1 = cinder_utils.get_qos(self.cinder, qos_settings=qos_settings) self.assertIsNotNone(qos1) validation_utils.objects_equivalent(self.qos, qos1) - qos2 = cinder_utils.get_qos(self.cinder, qos_name=qos_settings.name) + qos2 = cinder_utils.get_qos_by_id(self.cinder, qos1.id) self.assertIsNotNone(qos2) validation_utils.objects_equivalent(self.qos, qos2) @@ -139,8 +138,7 @@ class CinderUtilsQoSTests(OSComponentTestCase): Tests the cinder_utils.create_qos() """ qos_settings = QoSSettings(name=self.qos_name, consumer=Consumer.both) - self.qos = cinder_utils.create_qos( - self.cinder, qos_settings) + self.qos = cinder_utils.create_qos(self.cinder, qos_settings) self.assertIsNotNone(self.qos) self.assertEqual(self.qos_name, self.qos.name) @@ -152,3 +150,301 @@ class CinderUtilsQoSTests(OSComponentTestCase): cinder_utils.delete_qos(self.cinder, self.qos) self.assertIsNone(cinder_utils.get_qos( self.cinder, qos_settings=qos_settings)) + + +class CinderUtilsSimpleVolumeTypeTests(OSComponentTestCase): + """ + Tests the creation of a Volume Type without any external settings such as + QoS Specs or encryption + """ + + def setUp(self): + """ + Instantiates the CreateVolume object that is responsible for + downloading and creating an OS volume file within OpenStack + """ + guid = uuid.uuid4() + volume_type_name = self.__class__.__name__ + '-' + str(guid) + self.volume_type_settings = VolumeTypeSettings(name=volume_type_name) + self.volume_type = None + self.cinder = cinder_utils.cinder_client(self.os_creds) + + def tearDown(self): + """ + Cleans the remote OpenStack objects + """ + if self.volume_type: + try: + cinder_utils.delete_volume_type(self.cinder, self.volume_type) + except NotFound: + pass + + def test_create_simple_volume_type(self): + """ + Tests the cinder_utils.create_volume_type(), get_volume_type(), and + get_volume_type_by_id() + """ + self.volume_type = cinder_utils.create_volume_type( + self.cinder, self.volume_type_settings) + self.assertIsNotNone(self.volume_type) + self.assertEqual(self.volume_type_settings.name, self.volume_type.name) + + volume_type1 = cinder_utils.get_volume_type( + self.cinder, volume_type_settings=self.volume_type_settings) + self.assertEquals(self.volume_type, volume_type1) + self.assertEquals(self.volume_type_settings.public, + volume_type1.public) + + volume_type2 = cinder_utils.get_volume_type_by_id( + self.cinder, volume_type1.id) + self.assertEquals(self.volume_type, volume_type2) + self.assertIsNone(self.volume_type.encryption) + + def test_create_delete_volume_type(self): + """ + Primarily tests the cinder_utils.delete_volume() + """ + self.volume_type = cinder_utils.create_volume_type( + self.cinder, self.volume_type_settings) + self.assertIsNotNone(self.volume_type) + self.assertEqual(self.volume_type_settings.name, self.volume_type.name) + + volume_type = cinder_utils.get_volume_type( + self.cinder, volume_type_settings=self.volume_type_settings) + self.assertIsNotNone(volume_type) + validation_utils.objects_equivalent(self.volume_type, volume_type) + self.assertIsNone(self.volume_type.encryption) + + cinder_utils.delete_volume_type(self.cinder, self.volume_type) + self.assertIsNone(cinder_utils.get_volume_type_by_id( + self.cinder, self.volume_type.id)) + + +class CinderUtilsAddEncryptionTests(OSComponentTestCase): + """ + Tests the creation of an encryption and association to and existing + VolumeType object + """ + + def setUp(self): + """ + Instantiates the CreateVolume object that is responsible for + downloading and creating an OS volume file within OpenStack + """ + guid = uuid.uuid4() + self.encryption_name = self.__class__.__name__ + '-' + str(guid) + self.encryption = None + + self.cinder = cinder_utils.cinder_client(self.os_creds) + + volume_type_name = self.__class__.__name__ + '-' + str(guid) + '-type' + self.volume_type = cinder_utils.create_volume_type( + self.cinder, VolumeTypeSettings(name=volume_type_name)) + + def tearDown(self): + """ + Cleans the remote OpenStack objects + """ + if self.encryption: + try: + cinder_utils.delete_volume_type_encryption( + self.cinder, self.volume_type) + except NotFound: + pass + + if self.volume_type: + try: + cinder_utils.delete_volume_type(self.cinder, self.volume_type) + except NotFound: + pass + + def test_create_simple_encryption(self): + """ + Tests the cinder_utils.create_volume_encryption(), + get_volume_encryption(), and get_volume_encryption_by_id() + """ + encryption_settings = VolumeTypeEncryptionSettings( + name=self.encryption_name, provider_class='foo', + control_location=ControlLocation.front_end) + self.encryption = cinder_utils.create_volume_encryption( + self.cinder, self.volume_type, encryption_settings) + self.assertIsNotNone(self.encryption) + self.assertEqual('foo', self.encryption.provider) + self.assertEqual(ControlLocation.front_end.value, + self.encryption.control_location) + + encryption1 = cinder_utils.get_volume_encryption_by_type( + self.cinder, self.volume_type) + self.assertEquals(self.encryption, encryption1) + + def test_create_delete_encryption(self): + """ + Primarily tests the cinder_utils.delete_volume_type_encryption() + """ + encryption_settings = VolumeTypeEncryptionSettings( + name=self.encryption_name, provider_class='LuksEncryptor', + control_location=ControlLocation.back_end) + self.encryption = cinder_utils.create_volume_encryption( + self.cinder, self.volume_type, encryption_settings) + self.assertIsNotNone(self.encryption) + self.assertEqual('LuksEncryptor', self.encryption.provider) + self.assertEqual(ControlLocation.back_end.value, + self.encryption.control_location) + + encryption1 = cinder_utils.get_volume_encryption_by_type( + self.cinder, self.volume_type) + self.assertEquals(self.encryption, encryption1) + + cinder_utils.delete_volume_type_encryption( + self.cinder, self.volume_type) + + encryption2 = cinder_utils.get_volume_encryption_by_type( + self.cinder, self.volume_type) + self.assertIsNone(encryption2) + + def test_create_with_all_attrs(self): + """ + Tests the cinder_utils.create_volume_encryption() with all valid + settings + """ + encryption_settings = VolumeTypeEncryptionSettings( + name=self.encryption_name, provider_class='foo', + cipher='bar', control_location=ControlLocation.back_end, + key_size=1) + self.encryption = cinder_utils.create_volume_encryption( + self.cinder, self.volume_type, encryption_settings) + self.assertIsNotNone(self.encryption) + self.assertEqual('foo', self.encryption.provider) + self.assertEqual('bar', self.encryption.cipher) + self.assertEqual(1, self.encryption.key_size) + self.assertEqual(ControlLocation.back_end.value, + self.encryption.control_location) + + encryption1 = cinder_utils.get_volume_encryption_by_type( + self.cinder, self.volume_type) + self.assertEquals(self.encryption, encryption1) + + def test_create_bad_key_size(self): + """ + Tests the cinder_utils.create_volume_encryption() raises an exception + when the provider class does not exist + """ + encryption_settings = VolumeTypeEncryptionSettings( + name=self.encryption_name, provider_class='foo', + cipher='bar', control_location=ControlLocation.back_end, + key_size=-1) + + with self.assertRaises(BadRequest): + self.encryption = cinder_utils.create_volume_encryption( + self.cinder, self.volume_type, encryption_settings) + + +class CinderUtilsVolumeTypeCompleteTests(OSComponentTestCase): + """ + Tests to ensure that a volume type can have a QoS Spec added to it + """ + + def setUp(self): + """ + Creates objects for testing cinder_utils.py + """ + guid = uuid.uuid4() + self.qos_name = self.__class__.__name__ + '-' + str(guid) + '-qos' + self.vol_type_name = self.__class__.__name__ + '-' + str(guid) + self.specs = {'foo': 'bar'} + self.cinder = cinder_utils.cinder_client(self.os_creds) + qos_settings = QoSSettings(name=self.qos_name, specs=self.specs, + consumer=Consumer.both) + self.qos = cinder_utils.create_qos(self.cinder, qos_settings) + self.volume_type = None + + def tearDown(self): + """ + Cleans the remote OpenStack objects + """ + if self.volume_type: + if self.volume_type.encryption: + try: + cinder_utils.delete_volume_type_encryption( + self.cinder, self.volume_type) + except NotFound: + pass + try: + cinder_utils.delete_volume_type(self.cinder, self.volume_type) + except NotFound: + pass + + if self.qos: + try: + cinder_utils.delete_qos(self.cinder, self.qos) + except NotFound: + pass + + def test_create_with_encryption(self): + """ + Tests the cinder_utils.create_volume_type() where encryption has been + configured + """ + encryption_settings = VolumeTypeEncryptionSettings( + name='foo', provider_class='bar', + control_location=ControlLocation.back_end) + volume_type_settings = VolumeTypeSettings( + name=self.vol_type_name, encryption=encryption_settings) + self.volume_type = cinder_utils.create_volume_type( + self.cinder, volume_type_settings) + + vol_encrypt = cinder_utils.get_volume_encryption_by_type( + self.cinder, self.volume_type) + self.assertIsNotNone(vol_encrypt) + self.assertIsNone(self.volume_type.qos_spec) + self.assertEqual(self.volume_type.encryption, vol_encrypt) + self.assertEqual(self.volume_type.id, vol_encrypt.volume_type_id) + + def test_create_with_qos(self): + """ + Tests the cinder_utils.create_volume_type() with an associated QoS Spec + """ + volume_type_settings = VolumeTypeSettings( + name=self.vol_type_name, qos_spec_name=self.qos_name) + self.volume_type = cinder_utils.create_volume_type( + self.cinder, volume_type_settings) + + self.assertIsNotNone(self.volume_type) + self.assertIsNone(self.volume_type.encryption) + self.assertIsNotNone(self.volume_type.qos_spec) + self.assertEqual(self.qos.id, self.volume_type.qos_spec.id) + + def test_create_with_invalid_qos(self): + """ + Tests the cinder_utils.create_volume_type() when the QoS Spec name + does not exist + """ + volume_type_settings = VolumeTypeSettings( + name=self.vol_type_name, qos_spec_name='foo') + + self.volume_type = cinder_utils.create_volume_type( + self.cinder, volume_type_settings) + + self.assertIsNone(self.volume_type.qos_spec) + + def test_create_with_qos_and_encryption(self): + """ + Tests the cinder_utils.create_volume_type() with encryption and an + associated QoS Spec + """ + encryption_settings = VolumeTypeEncryptionSettings( + name='foo', provider_class='bar', + control_location=ControlLocation.back_end) + volume_type_settings = VolumeTypeSettings( + name=self.vol_type_name, qos_spec_name=self.qos_name, + encryption=encryption_settings) + self.volume_type = cinder_utils.create_volume_type( + self.cinder, volume_type_settings) + + self.assertIsNotNone(self.volume_type) + vol_encrypt = cinder_utils.get_volume_encryption_by_type( + self.cinder, self.volume_type) + self.assertIsNotNone(vol_encrypt) + self.assertEqual(self.volume_type.id, vol_encrypt.volume_type_id) + self.assertIsNotNone(self.volume_type.qos_spec) + self.assertEqual(self.qos.id, self.volume_type.qos_spec.id) diff --git a/snaps/test_suite_builder.py b/snaps/test_suite_builder.py index b71fdf1..a1b72aa 100644 --- a/snaps/test_suite_builder.py +++ b/snaps/test_suite_builder.py @@ -32,7 +32,9 @@ from snaps.domain.test.stack_tests import ( from snaps.domain.test.user_tests import UserDomainObjectTests from snaps.domain.test.vm_inst_tests import ( VmInstDomainObjectTests, FloatingIpDomainObjectTests) -from snaps.domain.test.volume_tests import QoSSpecDomainObjectTests +from snaps.domain.test.volume_tests import ( + QoSSpecDomainObjectTests, VolumeTypeDomainObjectTests, + VolumeTypeEncryptionObjectTests) from snaps.openstack.tests.conf.os_credentials_tests import ( ProxySettingsUnitTests, OSCredsUnitTests) from snaps.openstack.tests.create_flavor_tests import ( @@ -55,8 +57,8 @@ from snaps.openstack.tests.create_network_tests import ( from snaps.openstack.tests.create_project_tests import ( CreateProjectSuccessTests, ProjectSettingsUnitTests, CreateProjectUserTests) -from snaps.openstack.tests.create_qos_tests import (QoSSettingsUnitTests, - CreateQoSTests) +from snaps.openstack.tests.create_qos_tests import ( + QoSSettingsUnitTests, CreateQoSTests) from snaps.openstack.tests.create_router_tests import ( CreateRouterSuccessTests, CreateRouterNegativeTests, RouterSettingsUnitTests) @@ -68,10 +70,14 @@ from snaps.openstack.tests.create_stack_tests import ( CreateComplexStackTests) from snaps.openstack.tests.create_user_tests import ( UserSettingsUnitTests, CreateUserSuccessTests) +from snaps.openstack.tests.create_volume_type_tests import ( + VolumeTypeSettingsUnitTests, CreateSimpleVolumeTypeSuccessTests, + CreateVolumeTypeComplexTests) from snaps.openstack.tests.os_source_file_test import ( OSComponentTestCase, OSIntegrationTestCase) -from snaps.openstack.utils.tests.cinder_utils_tests import (CinderSmokeTests, - CinderUtilsQoSTests) +from snaps.openstack.utils.tests.cinder_utils_tests import ( + CinderSmokeTests, CinderUtilsQoSTests, CinderUtilsSimpleVolumeTypeTests, + CinderUtilsAddEncryptionTests, CinderUtilsVolumeTypeCompleteTests) from snaps.openstack.utils.tests.glance_utils_tests import ( GlanceSmokeTests, GlanceUtilsTests) from snaps.openstack.utils.tests.heat_utils_tests import ( @@ -169,6 +175,10 @@ def add_unit_tests(suite): suite.addTest(unittest.TestLoader().loadTestsFromTestCase( StackSettingsUnitTests)) suite.addTest(unittest.TestLoader().loadTestsFromTestCase( + VolumeTypeDomainObjectTests)) + suite.addTest(unittest.TestLoader().loadTestsFromTestCase( + VolumeTypeEncryptionObjectTests)) + suite.addTest(unittest.TestLoader().loadTestsFromTestCase( QoSSpecDomainObjectTests)) suite.addTest(unittest.TestLoader().loadTestsFromTestCase( VmInstDomainObjectTests)) @@ -176,6 +186,8 @@ def add_unit_tests(suite): FloatingIpDomainObjectTests)) suite.addTest(unittest.TestLoader().loadTestsFromTestCase( QoSSettingsUnitTests)) + suite.addTest(unittest.TestLoader().loadTestsFromTestCase( + VolumeTypeSettingsUnitTests)) def add_openstack_client_tests(suite, os_creds, ext_net_name, @@ -299,6 +311,18 @@ def add_openstack_api_tests(suite, os_creds, ext_net_name, use_keystone=True, CinderUtilsQoSTests, os_creds=os_creds, ext_net_name=ext_net_name, log_level=log_level, image_metadata=image_metadata)) + suite.addTest(OSComponentTestCase.parameterize( + CinderUtilsSimpleVolumeTypeTests, os_creds=os_creds, + ext_net_name=ext_net_name, log_level=log_level, + image_metadata=image_metadata)) + suite.addTest(OSComponentTestCase.parameterize( + CinderUtilsAddEncryptionTests, os_creds=os_creds, + ext_net_name=ext_net_name, log_level=log_level, + image_metadata=image_metadata)) + suite.addTest(OSComponentTestCase.parameterize( + CinderUtilsVolumeTypeCompleteTests, os_creds=os_creds, + ext_net_name=ext_net_name, log_level=log_level, + image_metadata=image_metadata)) def add_openstack_integration_tests(suite, os_creds, ext_net_name, @@ -383,6 +407,16 @@ def add_openstack_integration_tests(suite, os_creds, ext_net_name, ext_net_name=ext_net_name, use_keystone=use_keystone, flavor_metadata=flavor_metadata, image_metadata=image_metadata, log_level=log_level)) + suite.addTest(OSIntegrationTestCase.parameterize( + CreateSimpleVolumeTypeSuccessTests, os_creds=os_creds, + ext_net_name=ext_net_name, use_keystone=use_keystone, + flavor_metadata=flavor_metadata, image_metadata=image_metadata, + log_level=log_level)) + suite.addTest(OSIntegrationTestCase.parameterize( + CreateVolumeTypeComplexTests, os_creds=os_creds, + ext_net_name=ext_net_name, use_keystone=use_keystone, + flavor_metadata=flavor_metadata, image_metadata=image_metadata, + log_level=log_level)) # VM Instances suite.addTest(OSIntegrationTestCase.parameterize( |