rally-openstack/tests/unit/common/test_validators.py
Marian Gasparovic e199f7bc38 test designate resolving from VM
Change-Id: I7e35aa387a6c488ffc6532a1e5d3eb9387dc7963
2020-04-22 22:13:49 +02:00

1049 lines
41 KiB
Python

# Copyright 2017: Mirantis 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 copy
import ddt
from unittest import mock
from glanceclient import exc as glance_exc
from novaclient import exceptions as nova_exc
from rally import exceptions
from rally_openstack.common import consts
from rally_openstack.common import validators
from tests.unit import test
PATH = "rally_openstack.common.validators"
context = {
"admin": mock.MagicMock(),
"users": [mock.MagicMock()],
}
config = dict(args={"image": {"id": "fake_id",
"min_ram": 10,
"size": 1024 ** 3,
"min_disk": 10.0 * (1024 ** 3),
"image_name": "foo_image"},
"flavor": {"id": "fake_flavor_id",
"name": "test"},
"foo_image": {"id": "fake_image_id"}
},
context={"images": {"image_name": "foo_image"},
"api_versions@openstack": mock.MagicMock(),
"zones": {"set_zone_in_network": True}}
)
@mock.patch("rally_openstack.task.contexts.keystone.roles.RoleGenerator")
def test_with_roles_ctx(mock_role_generator):
@validators.with_roles_ctx()
def func(config, context):
pass
config = {"contexts": {}}
context = {"admin": {"credential": mock.MagicMock()},
"task": mock.MagicMock()}
func(config, context)
mock_role_generator().setup.assert_not_called()
config = {"contexts": {"roles": "admin"}}
func(config, context)
mock_role_generator().setup.assert_called_once_with()
class RequiredOpenStackValidatorTestCase(test.TestCase):
def validate(self):
validator = validators.RequiredOpenStackValidator(admin=True)
validator.validate(
{"platforms": {"openstack": {"admin": "foo"}}}, {}, None, None)
validator = validators.RequiredOpenStackValidator(users=True)
validator.validate(
{"platforms": {"openstack": {"admin": "foo"}}}, {}, None, None)
validator = validators.RequiredOpenStackValidator(users=True)
validator.validate(
{"platforms": {"openstack": {"users": ["foo"]}}}, {}, None, None)
def test_validate_failed(self):
# case #1: wrong configuration of validator
validator = validators.RequiredOpenStackValidator()
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, {}, {}, None, None)
self.assertEqual(
"You should specify admin=True or users=True or both.",
e.message)
# case #2: admin is not present
validator = validators.RequiredOpenStackValidator(admin=True)
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate,
{"platforms": {"openstack": {}}}, {}, None, None)
self.assertEqual("No admin credentials for openstack",
e.message)
# case #3: users are not present
validator = validators.RequiredOpenStackValidator(users=True)
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate,
{"platforms": {"openstack": {}}}, {}, None, None)
self.assertEqual("No user credentials for openstack",
e.message)
@ddt.ddt
class ImageExistsValidatorTestCase(test.TestCase):
def setUp(self):
super(ImageExistsValidatorTestCase, self).setUp()
self.validator = validators.ImageExistsValidator("image", True)
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
@ddt.unpack
@ddt.data(
{"param_name": "fake_param", "nullable": True, "err_msg": None},
{"param_name": "fake_param", "nullable": False,
"err_msg": "Parameter fake_param is not specified."},
{"param_name": "image", "nullable": True, "err_msg": None},
)
def test_validator(self, param_name, nullable, err_msg, ex=False):
validator = validators.ImageExistsValidator(param_name,
nullable)
clients = self.context["users"][0].clients.return_value
clients.glance().images.get = mock.Mock()
if ex:
clients.glance().images.get.side_effect = ex
if err_msg:
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, self.config, None, None)
self.assertEqual(err_msg, e.message)
else:
result = validator.validate(self.config, self.context, None,
None)
self.assertIsNone(result)
def test_validator_image_from_context(self):
config = {
"args": {"image": {"regex": r"^foo$"}},
"contexts": {"images": {"image_name": "foo"}}}
self.validator.validate(self.context, config, None, None)
@mock.patch("%s.openstack_types.GlanceImage" % PATH)
def test_validator_image_not_in_context(self, mock_glance_image):
mock_glance_image.return_value.pre_process.return_value = "image_id"
config = {
"args": {"image": "fake_image"},
"contexts": {
"images": {"fake_image_name": "foo"}}}
clients = self.context[
"users"][0]["credential"].clients.return_value
clients.glance().images.get = mock.Mock()
result = self.validator.validate(self.context, config, None, None)
self.assertIsNone(result)
mock_glance_image.assert_called_once_with(
context={"admin": {
"credential": self.context["users"][0]["credential"]}})
mock_glance_image.return_value.pre_process.assert_called_once_with(
config["args"]["image"], config={})
clients.glance().images.get.assert_called_with("image_id")
exs = [exceptions.InvalidScenarioArgument(),
glance_exc.HTTPNotFound()]
for ex in exs:
clients.glance().images.get.side_effect = ex
e = self.assertRaises(
validators.validation.ValidationError,
self.validator.validate, self.context, config, None, None)
self.assertEqual("Image 'fake_image' not found", e.message)
@ddt.ddt
class ExternalNetworkExistsValidatorTestCase(test.TestCase):
def setUp(self):
super(ExternalNetworkExistsValidatorTestCase, self).setUp()
self.validator = validators.ExternalNetworkExistsValidator("net")
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
@ddt.unpack
@ddt.data(
{"foo_conf": {}},
{"foo_conf": {"args": {"net": "custom"}}},
{"foo_conf": {"args": {"net": "non_exist"}},
"err_msg": "External (floating) network with name non_exist"
" not found by user {}. Available networks:"
" [{}, {}]"},
{"foo_conf": {"args": {"net": "custom"}},
"net1_name": {"name": {"net": "public"}},
"net2_name": {"name": {"net": "custom"}},
"err_msg": "External (floating) network with name custom"
" not found by user {}. Available networks:"
" [{}, {}]"}
)
def test_validator(self, foo_conf, net1_name="public", net2_name="custom",
err_msg=""):
user = self.context["users"][0]
net1 = {"name": net1_name, "router:external": True}
net2 = {"name": net2_name, "router:external": True}
user["credential"].clients().neutron().list_networks.return_value = {
"networks": [net1, net2]}
if err_msg:
e = self.assertRaises(
validators.validation.ValidationError,
self.validator.validate, self.context, foo_conf,
None, None)
self.assertEqual(
err_msg.format(user["credential"].username, net1, net2),
e.message)
else:
result = self.validator.validate(self.context, foo_conf,
None, None)
self.assertIsNone(result, "Unexpected result '%s'" % result)
@ddt.ddt
class RequiredNeutronExtensionsValidatorTestCase(test.TestCase):
def setUp(self):
super(RequiredNeutronExtensionsValidatorTestCase, self).setUp()
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
def test_validator(self):
validator = validators.RequiredNeutronExtensionsValidator(
"existing_extension")
clients = self.context["users"][0]["credential"].clients()
clients.neutron().list_extensions.return_value = {
"extensions": [{"alias": "existing_extension"}]}
validator.validate(self.context, {}, None, None)
def test_validator_failed(self):
err_msg = "Neutron extension absent_extension is not configured"
validator = validators.RequiredNeutronExtensionsValidator(
"absent_extension")
clients = self.context["users"][0]["credential"].clients()
clients.neutron().list_extensions.return_value = {
"extensions": [{"alias": "existing_extension"}]}
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, {}, None, None)
self.assertEqual(err_msg, e.message)
class FlavorExistsValidatorTestCase(test.TestCase):
def setUp(self):
super(FlavorExistsValidatorTestCase, self).setUp()
self.validator = validators.FlavorExistsValidator(
param_name="foo_flavor")
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
def test__get_validated_flavor_wrong_value_in_config(self):
e = self.assertRaises(
validators.validation.ValidationError,
self.validator._get_validated_flavor, self.config,
mock.MagicMock(), "foo_flavor")
self.assertEqual("Parameter foo_flavor is not specified.",
e.message)
@mock.patch("%s.openstack_types.Flavor" % PATH)
def test__get_validated_flavor(self, mock_flavor):
mock_flavor.return_value.pre_process.return_value = "flavor_id"
clients = mock.Mock()
clients.nova().flavors.get.return_value = "flavor"
result = self.validator._get_validated_flavor(self.config,
clients,
"flavor")
self.assertEqual("flavor", result)
mock_flavor.assert_called_once_with(
context={"admin": {"credential": clients.credential}}
)
mock_flavor_obj = mock_flavor.return_value
mock_flavor_obj.pre_process.assert_called_once_with(
self.config["args"]["flavor"], config={})
clients.nova().flavors.get.assert_called_once_with(flavor="flavor_id")
mock_flavor_obj.pre_process.reset_mock()
clients.side_effect = exceptions.InvalidScenarioArgument("")
result = self.validator._get_validated_flavor(
self.config, clients, "flavor")
self.assertEqual("flavor", result)
mock_flavor_obj.pre_process.assert_called_once_with(
self.config["args"]["flavor"], config={})
clients.nova().flavors.get.assert_called_with(flavor="flavor_id")
@mock.patch("%s.openstack_types.Flavor" % PATH)
def test__get_validated_flavor_not_found(self, mock_flavor):
mock_flavor.return_value.pre_process.return_value = "flavor_id"
clients = mock.MagicMock()
clients.nova().flavors.get.side_effect = nova_exc.NotFound("")
e = self.assertRaises(
validators.validation.ValidationError,
self.validator._get_validated_flavor,
self.config, clients, "flavor")
self.assertEqual("Flavor '%s' not found" %
self.config["args"]["flavor"],
e.message)
mock_flavor_obj = mock_flavor.return_value
mock_flavor_obj.pre_process.assert_called_once_with(
self.config["args"]["flavor"], config={})
@mock.patch("%s.types.obj_from_name" % PATH)
@mock.patch("%s.flavors_ctx.FlavorConfig" % PATH)
def test__get_flavor_from_context(self, mock_flavor_config,
mock_obj_from_name):
config = {
"contexts": {"images": {"fake_parameter_name": "foo_image"}}}
e = self.assertRaises(
validators.validation.ValidationError,
self.validator._get_flavor_from_context,
config, "foo_flavor")
self.assertEqual("No flavors context", e.message)
config = {"contexts": {"images": {"fake_parameter_name": "foo_image"},
"flavors": [{"flavor1": "fake_flavor1"}]}}
result = self.validator._get_flavor_from_context(config, "foo_flavor")
self.assertEqual("<context flavor: %s>" % result.name, result.id)
def test_validate(self):
expected_e = validators.validation.ValidationError("fpp")
self.validator._get_validated_flavor = mock.Mock(
side_effect=expected_e)
config = {}
ctx = mock.MagicMock()
actual_e = self.assertRaises(
validators.validation.ValidationError,
self.validator.validate, ctx, config, None, None)
self.assertEqual(expected_e, actual_e)
self.validator._get_validated_flavor.assert_called_once_with(
config=config,
clients=ctx["users"][0]["credential"].clients(),
param_name=self.validator.param_name)
@ddt.ddt
class ImageValidOnFlavorValidatorTestCase(test.TestCase):
def setUp(self):
super(ImageValidOnFlavorValidatorTestCase, self).setUp()
self.validator = validators.ImageValidOnFlavorValidator("foo_flavor",
"image")
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
@ddt.data(
{"validate_disk": True, "flavor_disk": True},
{"validate_disk": False, "flavor_disk": True},
{"validate_disk": False, "flavor_disk": False}
)
@ddt.unpack
def test_validate(self, validate_disk, flavor_disk):
validator = validators.ImageValidOnFlavorValidator(
flavor_param="foo_flavor",
image_param="image",
fail_on_404_image=False,
validate_disk=validate_disk)
min_ram = 2048
disk = 10
fake_image = {"min_ram": min_ram,
"size": disk * (1024 ** 3),
"min_disk": disk}
fake_flavor = mock.Mock(disk=None, ram=min_ram * 2)
if flavor_disk:
fake_flavor.disk = disk * 2
validator._get_validated_flavor = mock.Mock(
return_value=fake_flavor)
# case 1: no image, but it is ok, since fail_on_404_image is False
validator._get_validated_image = mock.Mock(
side_effect=validators.validation.ValidationError("!!!"))
validator.validate(self.context, {}, None, None)
# case 2: there is an image
validator._get_validated_image = mock.Mock(
return_value=fake_image)
validator.validate(self.context, {}, None, None)
# case 3: check caching of the flavor
self.context["users"].append(self.context["users"][0])
validator._get_validated_image.reset_mock()
validator._get_validated_flavor.reset_mock()
validator.validate(self.context, {}, None, None)
self.assertEqual(1, validator._get_validated_flavor.call_count)
self.assertEqual(2, validator._get_validated_image.call_count)
def test_validate_failed(self):
validator = validators.ImageValidOnFlavorValidator(
flavor_param="foo_flavor",
image_param="image",
fail_on_404_image=True,
validate_disk=True)
min_ram = 2048
disk = 10
fake_flavor = mock.Mock(disk=disk, ram=min_ram)
fake_flavor.id = "flavor_id"
validator._get_validated_flavor = mock.Mock(
return_value=fake_flavor)
# case 1: there is no image and fail_on_404_image flag is True
expected_e = validators.validation.ValidationError("!!!")
validator._get_validated_image = mock.Mock(
side_effect=expected_e)
actual_e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, {}, None, None
)
self.assertEqual(expected_e, actual_e)
# case 2: there is no right flavor
expected_e = KeyError("Ooops")
validator._get_validated_flavor.side_effect = expected_e
actual_e = self.assertRaises(
KeyError,
validator.validate, self.context, {}, None, None
)
self.assertEqual(expected_e, actual_e)
# case 3: ram of a flavor is less than min_ram of an image
validator._get_validated_flavor = mock.Mock(
return_value=fake_flavor)
fake_image = {"min_ram": min_ram * 2, "id": "image_id"}
validator._get_validated_image = mock.Mock(
return_value=fake_image)
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, {}, None, None
)
self.assertEqual(
"The memory size for flavor 'flavor_id' is too small for "
"requested image 'image_id'.", e.message)
# case 4: disk of a flavor is less than size of an image
fake_image = {"min_ram": min_ram / 2.0,
"size": disk * (1024 ** 3) * 3,
"id": "image_id"}
validator._get_validated_image = mock.Mock(
return_value=fake_image)
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, {}, None, None
)
self.assertEqual(
"The disk size for flavor 'flavor_id' is too small for "
"requested image 'image_id'.", e.message)
# case 5: disk of a flavor is less than size of an image
fake_image = {"min_ram": min_ram,
"size": disk * (1024 ** 3),
"min_disk": disk * 2,
"id": "image_id"}
validator._get_validated_image = mock.Mock(
return_value=fake_image)
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, {}, None, None
)
self.assertEqual(
"The minimal disk size for flavor 'flavor_id' is too small for "
"requested image 'image_id'.", e.message)
# case 6: _get_validated_image raises an unexpected error,
# fail_on_404_image=False should not work in this case
expected_e = KeyError("Foo!")
validator = validators.ImageValidOnFlavorValidator(
flavor_param="foo_flavor",
image_param="image",
fail_on_404_image=False,
validate_disk=True)
validator._get_validated_image = mock.Mock(
side_effect=expected_e)
validator._get_validated_flavor = mock.Mock()
actual_e = self.assertRaises(
KeyError,
validator.validate, self.context, {}, None, None
)
self.assertEqual(expected_e, actual_e)
@mock.patch("%s.openstack_types.GlanceImage" % PATH)
def test__get_validated_image(self, mock_glance_image):
mock_glance_image.return_value.pre_process.return_value = "image_id"
image = {
"size": 0,
"min_ram": 0,
"min_disk": 0
}
# Get image name from context
result = self.validator._get_validated_image({
"args": {
"image": {"regex": r"^foo$"}},
"contexts": {
"images": {"image_name": "foo"}}},
mock.Mock(), "image")
self.assertEqual(image, result)
clients = mock.Mock()
clients.glance().images.get().to_dict.return_value = {
"image": "image_id"}
image["image"] = "image_id"
result = self.validator._get_validated_image(self.config,
clients,
"image")
self.assertEqual(image, result)
mock_glance_image.assert_called_once_with(
context={"admin": {"credential": clients.credential}})
mock_glance_image.return_value.pre_process.assert_called_once_with(
config["args"]["image"], config={})
clients.glance().images.get.assert_called_with("image_id")
@mock.patch("%s.openstack_types.GlanceImage" % PATH)
def test__get_validated_image_incorrect_param(self, mock_glance_image):
mock_glance_image.return_value.pre_process.return_value = "image_id"
# Wrong 'param_name'
e = self.assertRaises(
validators.validation.ValidationError,
self.validator._get_validated_image, self.config,
mock.Mock(), "fake_param")
self.assertEqual("Parameter fake_param is not specified.",
e.message)
# 'image_name' is not in 'image_context'
image = {"id": "image_id", "size": 1024,
"min_ram": 256, "min_disk": 512}
clients = mock.Mock()
clients.glance().images.get().to_dict.return_value = image
config = {"args": {"image": "foo_image",
"context": {"images": {
"fake_parameter_name": "foo_image"}
}}
}
result = self.validator._get_validated_image(config, clients, "image")
self.assertEqual(image, result)
mock_glance_image.assert_called_once_with(
context={"admin": {"credential": clients.credential}})
mock_glance_image.return_value.pre_process.assert_called_once_with(
config["args"]["image"], config={})
clients.glance().images.get.assert_called_with("image_id")
@mock.patch("%s.openstack_types.GlanceImage" % PATH)
def test__get_validated_image_exceptions(self, mock_glance_image):
mock_glance_image.return_value.pre_process.return_value = "image_id"
clients = mock.Mock()
clients.glance().images.get.side_effect = glance_exc.HTTPNotFound("")
e = self.assertRaises(
validators.validation.ValidationError,
self.validator._get_validated_image,
config, clients, "image")
self.assertEqual("Image '%s' not found" % config["args"]["image"],
e.message)
mock_glance_image.assert_called_once_with(
context={"admin": {"credential": clients.credential}})
mock_glance_image.return_value.pre_process.assert_called_once_with(
config["args"]["image"], config={})
clients.glance().images.get.assert_called_with("image_id")
mock_glance_image.return_value.pre_process.reset_mock()
clients.side_effect = exceptions.InvalidScenarioArgument("")
e = self.assertRaises(
validators.validation.ValidationError,
self.validator._get_validated_image, config, clients, "image")
self.assertEqual("Image '%s' not found" % config["args"]["image"],
e.message)
mock_glance_image.return_value.pre_process.assert_called_once_with(
config["args"]["image"], config={})
clients.glance().images.get.assert_called_with("image_id")
class RequiredServicesValidatorTestCase(test.TestCase):
def setUp(self):
super(RequiredServicesValidatorTestCase, self).setUp()
self.validator = validators.RequiredServicesValidator([
consts.Service.KEYSTONE,
consts.Service.NOVA])
self.config = config
self.context = context
def test_validator(self):
self.config["context"]["api_versions@openstack"].get = mock.Mock(
return_value={consts.Service.KEYSTONE: "service_type"})
clients = self.context["admin"].get("credential").clients()
clients.services().values.return_value = [
consts.Service.KEYSTONE, consts.Service.NOVA,
consts.Service.NOVA_NET]
fake_service = mock.Mock(binary="nova-network", status="enabled")
clients.nova.services.list.return_value = [fake_service]
result = self.validator.validate(self.context, self.config,
None, None)
self.assertIsNone(result)
fake_service = mock.Mock(binary="keystone", status="enabled")
clients.nova.services.list.return_value = [fake_service]
result = self.validator.validate(self.context, self.config,
None, None)
self.assertIsNone(result)
fake_service = mock.Mock(binary="nova-network", status="disabled")
clients.nova.services.list.return_value = [fake_service]
result = self.validator.validate(self.context, self.config,
None, None)
self.assertIsNone(result)
def test_validator_wrong_service(self):
self.config["context"]["api_versions@openstack"].get = mock.Mock(
return_value={consts.Service.KEYSTONE: "service_type",
consts.Service.NOVA: "service_name"})
clients = self.context["admin"].get("credential").clients()
clients.services().values.return_value = [
consts.Service.KEYSTONE, consts.Service.NOVA]
validator = validators.RequiredServicesValidator([
consts.Service.KEYSTONE,
consts.Service.NOVA, "lol"])
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, {}, None, None)
expected_msg = ("'{0}' service is not available. Hint: If '{0}'"
" service has non-default service_type, try to setup"
" it via 'api_versions@openstack' context."
).format("lol")
self.assertEqual(expected_msg, e.message)
@ddt.ddt
class ValidateHeatTemplateValidatorTestCase(test.TestCase):
def setUp(self):
super(ValidateHeatTemplateValidatorTestCase, self).setUp()
self.validator = validators.ValidateHeatTemplateValidator(
"template_path1", "template_path2")
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
@ddt.data(
{"exception_msg": "Heat template validation failed on fake_path1. "
"Original error message: fake_msg."},
{"exception_msg": None}
)
@ddt.unpack
@mock.patch("%s.os.path.exists" % PATH,
return_value=True)
@mock.patch("rally_openstack.common.validators.open",
side_effect=mock.mock_open(), create=True)
def test_validate(self, mock_open, mock_exists, exception_msg):
clients = self.context["users"][0]["credential"].clients()
mock_open().__enter__().read.side_effect = ["fake_template1",
"fake_template2"]
heat_validator = mock.MagicMock()
if exception_msg:
heat_validator.side_effect = Exception("fake_msg")
clients.heat().stacks.validate = heat_validator
context = {"args": {"template_path1": "fake_path1",
"template_path2": "fake_path2"}}
if not exception_msg:
result = self.validator.validate(self.context, context, None, None)
heat_validator.assert_has_calls([
mock.call(template="fake_template1"),
mock.call(template="fake_template2")
])
mock_open.assert_has_calls([
mock.call("fake_path1", "r"),
mock.call("fake_path2", "r")
], any_order=True)
self.assertIsNone(result)
else:
e = self.assertRaises(
validators.validation.ValidationError,
self.validator.validate, self.context, context, None, None)
heat_validator.assert_called_once_with(
template="fake_template1")
self.assertEqual(
"Heat template validation failed on fake_path1."
" Original error message: fake_msg.", e.message)
def test_validate_missed_params(self):
validator = validators.ValidateHeatTemplateValidator(
params="fake_param")
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, self.config, None, None)
expected_msg = ("Path to heat template is not specified. Its needed "
"for heat template validation. Please check the "
"content of `fake_param` scenario argument.")
self.assertEqual(expected_msg, e.message)
@mock.patch("%s.os.path.exists" % PATH,
return_value=False)
def test_validate_file_not_found(self, mock_exists):
config = {"args": {"template_path1": "fake_path1",
"template_path2": "fake_path2"}}
e = self.assertRaises(
validators.validation.ValidationError,
self.validator.validate, self.context, config, None, None)
expected_msg = "No file found by the given path fake_path1"
self.assertEqual(expected_msg, e.message)
class RequiredCinderServicesValidatorTestCase(test.TestCase):
def setUp(self):
super(RequiredCinderServicesValidatorTestCase, self).setUp()
self.context = copy.deepcopy(context)
self.config = copy.deepcopy(config)
def test_validate(self):
validator = validators.RequiredCinderServicesValidator(
"cinder_service")
fake_service = mock.Mock(binary="cinder_service", state="up")
clients = self.context["admin"]["credential"].clients()
clients.cinder().services.list.return_value = [fake_service]
result = validator.validate(self.context, self.config, None, None)
self.assertIsNone(result)
fake_service.state = "down"
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, self.config, None, None)
self.assertEqual("cinder_service service is not available",
e.message)
@ddt.ddt
class RequiredAPIVersionsValidatorTestCase(test.TestCase):
def setUp(self):
super(RequiredAPIVersionsValidatorTestCase, self).setUp()
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
def _get_keystone_v2_mock_client(self):
keystone = mock.Mock()
del keystone.projects
keystone.tenants = mock.Mock()
return keystone
def _get_keystone_v3_mock_client(self):
keystone = mock.Mock()
del keystone.tenants
keystone.projects = mock.Mock()
return keystone
def test_validate(self):
validator = validators.RequiredAPIVersionsValidator("keystone",
[2.0, 3])
clients = self.context["users"][0]["credential"].clients()
clients.keystone.return_value = self._get_keystone_v3_mock_client()
validator.validate(self.context, self.config, None, None)
clients.keystone.return_value = self._get_keystone_v2_mock_client()
validator.validate(self.context, self.config, None, None)
def test_validate_with_keystone_v2(self):
validator = validators.RequiredAPIVersionsValidator("keystone",
[2.0])
clients = self.context["users"][0]["credential"].clients()
clients.keystone.return_value = self._get_keystone_v2_mock_client()
validator.validate(self.context, self.config, None, None)
clients.keystone.return_value = self._get_keystone_v3_mock_client()
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, self.config, None, None)
self.assertEqual("Task was designed to be used with keystone V2.0, "
"but V3 is selected.", e.message)
def test_validate_with_keystone_v3(self):
validator = validators.RequiredAPIVersionsValidator("keystone",
[3])
clients = self.context["users"][0]["credential"].clients()
clients.keystone.return_value = self._get_keystone_v3_mock_client()
validator.validate(self.context, self.config, None, None)
clients.keystone.return_value = self._get_keystone_v2_mock_client()
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, self.config, None, None)
self.assertEqual("Task was designed to be used with keystone V3, "
"but V2.0 is selected.", e.message)
@ddt.unpack
@ddt.data(
{"nova": 2, "versions": [2], "err_msg": None},
{"nova": 3, "versions": [2],
"err_msg": "Task was designed to be used with nova V2, "
"but V3 is selected."},
{"nova": None, "versions": [2],
"err_msg": "Unable to determine the API version."},
{"nova": 2, "versions": [2, 3], "err_msg": None},
{"nova": 4, "versions": [2, 3],
"err_msg": "Task was designed to be used with nova V2, 3, "
"but V4 is selected."}
)
def test_validate_nova(self, nova, versions, err_msg):
validator = validators.RequiredAPIVersionsValidator("nova",
versions)
clients = self.context["users"][0]["credential"].clients()
clients.nova.choose_version.return_value = nova
config = {"contexts": {"api_versions@openstack": {}}}
if err_msg:
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, config, None, None)
self.assertEqual(err_msg, e.message)
else:
result = validator.validate(self.context, config, None, None)
self.assertIsNone(result)
@ddt.unpack
@ddt.data({"version": 2, "err_msg": None},
{"version": 3, "err_msg": "Task was designed to be used with "
"nova V3, but V2 is selected."})
def test_validate_context(self, version, err_msg):
validator = validators.RequiredAPIVersionsValidator("nova",
[version])
config = {
"contexts": {"api_versions@openstack": {"nova": {"version": 2}}}}
if err_msg:
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, config, None, None)
self.assertEqual(err_msg, e.message)
else:
result = validator.validate(self.context, config, None, None)
self.assertIsNone(result)
class VolumeTypeExistsValidatorTestCase(test.TestCase):
def setUp(self):
super(VolumeTypeExistsValidatorTestCase, self).setUp()
self.validator = validators.VolumeTypeExistsValidator("volume_type",
True)
self.config = copy.deepcopy(config)
self.context = copy.deepcopy(context)
def test_validator_without_ctx(self):
validator = validators.VolumeTypeExistsValidator("fake_param",
nullable=True)
clients = self.context["users"][0]["credential"].clients()
clients.cinder().volume_types.list.return_value = [mock.MagicMock()]
result = validator.validate(self.context, self.config, None, None)
self.assertIsNone(result, "Unexpected result")
def test_validator_without_ctx_failed(self):
validator = validators.VolumeTypeExistsValidator("fake_param",
nullable=False)
clients = self.context["users"][0]["credential"].clients()
clients.cinder().volume_types.list.return_value = [mock.MagicMock()]
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, self.context, self.config, None, None)
self.assertEqual(
"The parameter 'fake_param' is required and should not be empty.",
e.message)
def test_validate_with_ctx(self):
clients = self.context["users"][0]["credential"].clients()
clients.cinder().volume_types.list.return_value = []
ctx = {"args": {"volume_type": "fake_type"},
"contexts": {"volume_types": ["fake_type"]}}
result = self.validator.validate(self.context, ctx, None, None)
self.assertIsNone(result)
def test_validate_with_ctx_failed(self):
clients = self.context["users"][0]["credential"].clients()
clients.cinder().volume_types.list.return_value = []
config = {"args": {"volume_type": "fake_type"},
"contexts": {"volume_types": ["fake_type_2"]}}
e = self.assertRaises(
validators.validation.ValidationError,
self.validator.validate, self.context, config, None, None)
err_msg = ("Specified volume type fake_type not found for user {}. "
"List of available types: ['fake_type_2']")
fake_user = self.context["users"][0]
self.assertEqual(err_msg.format(fake_user), e.message)
@ddt.ddt
class WorkbookContainsWorkflowValidatorTestCase(test.TestCase):
@mock.patch("%s.yaml.safe_load" % PATH)
@mock.patch("%s.os.access" % PATH)
@mock.patch("%s.open" % PATH)
def test_validator(self, mock_open, mock_access, mock_safe_load):
mock_safe_load.return_value = {
"version": "2.0",
"name": "wb",
"workflows": {
"wf1": {
"type": "direct",
"tasks": {
"t1": {
"action": "std.noop"
}
}
}
}
}
validator = validators.WorkbookContainsWorkflowValidator(
workbook_param="definition", workflow_param="workflow_name")
config = {
"args": {
"definition": "fake_path1",
"workflow_name": "wf1"
}
}
result = validator.validate(None, config, None, None)
self.assertIsNone(result)
self.assertEqual(1, mock_open.called)
self.assertEqual(1, mock_access.called)
self.assertEqual(1, mock_safe_load.called)
@ddt.ddt
class RequiredContextConfigValidatorTestCase(test.TestCase):
def test_validator(self):
validator = validators.RequiredContextConfigValidator(
context_name="zones",
context_config={"set_zone_in_network": True})
cfg = {
"contexts": {
"users": {
"tenants": 1, "users_per_tenant": 1
},
"network": {
"dns_nameservers": ["8.8.8.8", "192.168.210.45"]
},
"zones": {"set_zone_in_network": True}
},
}
validator.validate({}, cfg, None, None)
def test_validator_context_not_in_contexts(self):
validator = validators.RequiredContextConfigValidator(
context_name="zones",
context_config={"set_zone_in_network": True})
cfg = {
"contexts": {
"users": {
"tenants": 1, "users_per_tenant": 1
},
"network": {
"dns_nameservers": ["8.8.8.8", "192.168.210.45"]
},
},
}
validator.validate({}, cfg, None, None)
def test_validator_failed(self):
validator = validators.RequiredContextConfigValidator(
context_name="zones",
context_config={"set_zone_in_network": True})
cfg = {
"contexts": {
"users": {
"tenants": 1, "users_per_tenant": 1
},
"network": {
"dns_nameservers": ["8.8.8.8", "192.168.210.45"]
},
"zones": {"set_zone_in_network": False}
},
}
e = self.assertRaises(
validators.validation.ValidationError,
validator.validate, {}, cfg, None, None)
self.assertEqual(
"The 'zones' context expects '{'set_zone_in_network': True}'",
e.message)