Merge "Add unit test for volume type in volume v1"
This commit is contained in:
commit
fb66a5b8f9
@ -474,3 +474,77 @@ class TestVolumev1(utils.TestCommand):
|
|||||||
endpoint=fakes.AUTH_URL,
|
endpoint=fakes.AUTH_URL,
|
||||||
token=fakes.AUTH_TOKEN,
|
token=fakes.AUTH_TOKEN,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class FakeType(object):
|
||||||
|
"""Fake one or more type."""
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def create_one_type(attrs=None, methods=None):
|
||||||
|
"""Create a fake type.
|
||||||
|
|
||||||
|
:param Dictionary attrs:
|
||||||
|
A dictionary with all attributes
|
||||||
|
:param Dictionary methods:
|
||||||
|
A dictionary with all methods
|
||||||
|
:return:
|
||||||
|
A FakeResource object with id, name, description, etc.
|
||||||
|
"""
|
||||||
|
attrs = attrs or {}
|
||||||
|
methods = methods or {}
|
||||||
|
|
||||||
|
# Set default attributes.
|
||||||
|
type_info = {
|
||||||
|
"id": 'type-id-' + uuid.uuid4().hex,
|
||||||
|
"name": 'type-name-' + uuid.uuid4().hex,
|
||||||
|
"description": 'type-description-' + uuid.uuid4().hex,
|
||||||
|
"extra_specs": {"foo": "bar"},
|
||||||
|
"is_public": True,
|
||||||
|
}
|
||||||
|
|
||||||
|
# Overwrite default attributes.
|
||||||
|
type_info.update(attrs)
|
||||||
|
|
||||||
|
volume_type = fakes.FakeResource(
|
||||||
|
info=copy.deepcopy(type_info),
|
||||||
|
methods=methods,
|
||||||
|
loaded=True)
|
||||||
|
return volume_type
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def create_types(attrs=None, count=2):
|
||||||
|
"""Create multiple fake types.
|
||||||
|
|
||||||
|
:param Dictionary attrs:
|
||||||
|
A dictionary with all attributes
|
||||||
|
:param int count:
|
||||||
|
The number of types to fake
|
||||||
|
:return:
|
||||||
|
A list of FakeResource objects faking the types
|
||||||
|
"""
|
||||||
|
volume_types = []
|
||||||
|
for i in range(0, count):
|
||||||
|
volume_type = FakeType.create_one_type(attrs)
|
||||||
|
volume_types.append(volume_type)
|
||||||
|
|
||||||
|
return volume_types
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_types(types=None, count=2):
|
||||||
|
"""Get an iterable MagicMock object with a list of faked types.
|
||||||
|
|
||||||
|
If types list is provided, then initialize the Mock object with the
|
||||||
|
list. Otherwise create one.
|
||||||
|
|
||||||
|
:param List types:
|
||||||
|
A list of FakeResource objects faking types
|
||||||
|
:param Integer count:
|
||||||
|
The number of types to be faked
|
||||||
|
:return
|
||||||
|
An iterable Mock object with side_effect set to a list of faked
|
||||||
|
types
|
||||||
|
"""
|
||||||
|
if types is None:
|
||||||
|
types = FakeType.create_types(count)
|
||||||
|
|
||||||
|
return mock.MagicMock(side_effect=types)
|
||||||
|
347
openstackclient/tests/unit/volume/v1/test_type.py
Normal file
347
openstackclient/tests/unit/volume/v1/test_type.py
Normal file
@ -0,0 +1,347 @@
|
|||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
#
|
||||||
|
|
||||||
|
import mock
|
||||||
|
from mock import call
|
||||||
|
|
||||||
|
from osc_lib import exceptions
|
||||||
|
from osc_lib import utils
|
||||||
|
|
||||||
|
from openstackclient.tests.unit import utils as tests_utils
|
||||||
|
from openstackclient.tests.unit.volume.v1 import fakes as volume_fakes
|
||||||
|
from openstackclient.volume.v1 import volume_type
|
||||||
|
|
||||||
|
|
||||||
|
class TestType(volume_fakes.TestVolumev1):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestType, self).setUp()
|
||||||
|
|
||||||
|
self.types_mock = self.app.client_manager.volume.volume_types
|
||||||
|
self.types_mock.reset_mock()
|
||||||
|
|
||||||
|
|
||||||
|
class TestTypeCreate(TestType):
|
||||||
|
|
||||||
|
columns = (
|
||||||
|
'description',
|
||||||
|
'id',
|
||||||
|
'is_public',
|
||||||
|
'name',
|
||||||
|
)
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestTypeCreate, self).setUp()
|
||||||
|
|
||||||
|
self.new_volume_type = volume_fakes.FakeType.create_one_type(
|
||||||
|
methods={'set_keys': {'myprop': 'myvalue'}}
|
||||||
|
)
|
||||||
|
self.data = (
|
||||||
|
self.new_volume_type.description,
|
||||||
|
self.new_volume_type.id,
|
||||||
|
True,
|
||||||
|
self.new_volume_type.name,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.types_mock.create.return_value = self.new_volume_type
|
||||||
|
# Get the command object to test
|
||||||
|
self.cmd = volume_type.CreateVolumeType(self.app, None)
|
||||||
|
|
||||||
|
def test_type_create(self):
|
||||||
|
arglist = [
|
||||||
|
self.new_volume_type.name,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
("name", self.new_volume_type.name),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
self.types_mock.create.assert_called_with(
|
||||||
|
self.new_volume_type.name,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, data)
|
||||||
|
|
||||||
|
|
||||||
|
class TestTypeDelete(TestType):
|
||||||
|
|
||||||
|
volume_types = volume_fakes.FakeType.create_types(count=2)
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestTypeDelete, self).setUp()
|
||||||
|
|
||||||
|
self.types_mock.get = volume_fakes.FakeType.get_types(
|
||||||
|
self.volume_types)
|
||||||
|
self.types_mock.delete.return_value = None
|
||||||
|
|
||||||
|
# Get the command object to mock
|
||||||
|
self.cmd = volume_type.DeleteVolumeType(self.app, None)
|
||||||
|
|
||||||
|
def test_type_delete(self):
|
||||||
|
arglist = [
|
||||||
|
self.volume_types[0].id
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
("volume_types", [self.volume_types[0].id])
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.types_mock.delete.assert_called_with(self.volume_types[0])
|
||||||
|
self.assertIsNone(result)
|
||||||
|
|
||||||
|
def test_delete_multiple_types(self):
|
||||||
|
arglist = []
|
||||||
|
for t in self.volume_types:
|
||||||
|
arglist.append(t.id)
|
||||||
|
verifylist = [
|
||||||
|
('volume_types', arglist),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
calls = []
|
||||||
|
for t in self.volume_types:
|
||||||
|
calls.append(call(t))
|
||||||
|
self.types_mock.delete.assert_has_calls(calls)
|
||||||
|
self.assertIsNone(result)
|
||||||
|
|
||||||
|
def test_delete_multiple_types_with_exception(self):
|
||||||
|
arglist = [
|
||||||
|
self.volume_types[0].id,
|
||||||
|
'unexist_type',
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('volume_types', arglist),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
find_mock_result = [self.volume_types[0], exceptions.CommandError]
|
||||||
|
with mock.patch.object(utils, 'find_resource',
|
||||||
|
side_effect=find_mock_result) as find_mock:
|
||||||
|
try:
|
||||||
|
self.cmd.take_action(parsed_args)
|
||||||
|
self.fail('CommandError should be raised.')
|
||||||
|
except exceptions.CommandError as e:
|
||||||
|
self.assertEqual('1 of 2 volume types failed to delete.',
|
||||||
|
str(e))
|
||||||
|
|
||||||
|
find_mock.assert_any_call(
|
||||||
|
self.types_mock, self.volume_types[0].id)
|
||||||
|
find_mock.assert_any_call(self.types_mock, 'unexist_type')
|
||||||
|
|
||||||
|
self.assertEqual(2, find_mock.call_count)
|
||||||
|
self.types_mock.delete.assert_called_once_with(
|
||||||
|
self.volume_types[0]
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestTypeList(TestType):
|
||||||
|
|
||||||
|
volume_types = volume_fakes.FakeType.create_types()
|
||||||
|
|
||||||
|
columns = (
|
||||||
|
"ID",
|
||||||
|
"Name"
|
||||||
|
)
|
||||||
|
columns_long = (
|
||||||
|
"ID",
|
||||||
|
"Name",
|
||||||
|
"Properties"
|
||||||
|
)
|
||||||
|
|
||||||
|
data = []
|
||||||
|
for t in volume_types:
|
||||||
|
data.append((
|
||||||
|
t.id,
|
||||||
|
t.name,
|
||||||
|
))
|
||||||
|
data_long = []
|
||||||
|
for t in volume_types:
|
||||||
|
data_long.append((
|
||||||
|
t.id,
|
||||||
|
t.name,
|
||||||
|
utils.format_dict(t.extra_specs),
|
||||||
|
))
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestTypeList, self).setUp()
|
||||||
|
|
||||||
|
self.types_mock.list.return_value = self.volume_types
|
||||||
|
# get the command to test
|
||||||
|
self.cmd = volume_type.ListVolumeType(self.app, None)
|
||||||
|
|
||||||
|
def test_type_list_without_options(self):
|
||||||
|
arglist = []
|
||||||
|
verifylist = [
|
||||||
|
("long", False),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
self.types_mock.list.assert_called_once_with()
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, list(data))
|
||||||
|
|
||||||
|
def test_type_list_with_options(self):
|
||||||
|
arglist = [
|
||||||
|
"--long",
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
("long", True),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
self.types_mock.list.assert_called_once_with()
|
||||||
|
self.assertEqual(self.columns_long, columns)
|
||||||
|
self.assertEqual(self.data_long, list(data))
|
||||||
|
|
||||||
|
|
||||||
|
class TestTypeSet(TestType):
|
||||||
|
|
||||||
|
volume_type = volume_fakes.FakeType.create_one_type(
|
||||||
|
methods={'set_keys': None})
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestTypeSet, self).setUp()
|
||||||
|
|
||||||
|
self.types_mock.get.return_value = self.volume_type
|
||||||
|
|
||||||
|
# Get the command object to test
|
||||||
|
self.cmd = volume_type.SetVolumeType(self.app, None)
|
||||||
|
|
||||||
|
def test_type_set_nothing(self):
|
||||||
|
arglist = [
|
||||||
|
self.volume_type.id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('volume_type', self.volume_type.id),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.assertIsNone(result)
|
||||||
|
|
||||||
|
def test_type_set_property(self):
|
||||||
|
arglist = [
|
||||||
|
'--property', 'myprop=myvalue',
|
||||||
|
self.volume_type.id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('property', {'myprop': 'myvalue'}),
|
||||||
|
('volume_type', self.volume_type.id),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
self.volume_type.set_keys.assert_called_once_with(
|
||||||
|
{'myprop': 'myvalue'})
|
||||||
|
self.assertIsNone(result)
|
||||||
|
|
||||||
|
|
||||||
|
class TestTypeShow(TestType):
|
||||||
|
|
||||||
|
columns = (
|
||||||
|
'description',
|
||||||
|
'id',
|
||||||
|
'is_public',
|
||||||
|
'name',
|
||||||
|
'properties',
|
||||||
|
)
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestTypeShow, self).setUp()
|
||||||
|
|
||||||
|
self.volume_type = volume_fakes.FakeType.create_one_type()
|
||||||
|
self.data = (
|
||||||
|
self.volume_type.description,
|
||||||
|
self.volume_type.id,
|
||||||
|
True,
|
||||||
|
self.volume_type.name,
|
||||||
|
utils.format_dict(self.volume_type.extra_specs)
|
||||||
|
)
|
||||||
|
|
||||||
|
self.types_mock.get.return_value = self.volume_type
|
||||||
|
|
||||||
|
# Get the command object to test
|
||||||
|
self.cmd = volume_type.ShowVolumeType(self.app, None)
|
||||||
|
|
||||||
|
def test_type_show(self):
|
||||||
|
arglist = [
|
||||||
|
self.volume_type.id
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
("volume_type", self.volume_type.id)
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
self.types_mock.get.assert_called_with(self.volume_type.id)
|
||||||
|
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, data)
|
||||||
|
|
||||||
|
|
||||||
|
class TestTypeUnset(TestType):
|
||||||
|
|
||||||
|
volume_type = volume_fakes.FakeType.create_one_type(
|
||||||
|
methods={'unset_keys': None})
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestTypeUnset, self).setUp()
|
||||||
|
|
||||||
|
self.types_mock.get.return_value = self.volume_type
|
||||||
|
|
||||||
|
# Get the command object to test
|
||||||
|
self.cmd = volume_type.UnsetVolumeType(self.app, None)
|
||||||
|
|
||||||
|
def test_type_unset(self):
|
||||||
|
arglist = [
|
||||||
|
'--property', 'property',
|
||||||
|
'--property', 'multi_property',
|
||||||
|
self.volume_type.id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('property', ['property', 'multi_property']),
|
||||||
|
('volume_type', self.volume_type.id),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
self.volume_type.unset_keys.assert_called_once_with(
|
||||||
|
['property', 'multi_property'])
|
||||||
|
self.assertIsNone(result)
|
||||||
|
|
||||||
|
def test_type_unset_failed_with_missing_volume_type_argument(self):
|
||||||
|
arglist = [
|
||||||
|
'--property', 'property',
|
||||||
|
'--property', 'multi_property',
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('property', ['property', 'multi_property']),
|
||||||
|
]
|
||||||
|
|
||||||
|
self.assertRaises(tests_utils.ParserException,
|
||||||
|
self.check_parser,
|
||||||
|
self.cmd,
|
||||||
|
arglist,
|
||||||
|
verifylist)
|
Loading…
Reference in New Issue
Block a user