Add cinder.CreateAndSetQos scenario

Create a qos, then Add/Update keys in qos specs.

Change-Id: I26fe73d1b4833ca23023698e22ee8387d358d786
This commit is contained in:
zhangzhang 2017-04-21 13:04:46 -04:00
parent dc2c7e49ce
commit 0993e8f42a
13 changed files with 226 additions and 32 deletions

View File

@ -1081,7 +1081,6 @@
CinderQos.create_and_list_qos: CinderQos.create_and_list_qos:
- -
args: args:
specs:
consumer: "both" consumer: "both"
write_iops_sec: "10" write_iops_sec: "10"
read_iops_sec: "1000" read_iops_sec: "1000"
@ -1100,7 +1099,6 @@
CinderQos.create_and_get_qos: CinderQos.create_and_get_qos:
- -
args: args:
specs:
consumer: "both" consumer: "both"
write_iops_sec: "10" write_iops_sec: "10"
read_iops_sec: "1000" read_iops_sec: "1000"
@ -1169,3 +1167,24 @@
sla: sla:
failure_rate: failure_rate:
max: 0 max: 0
CinderQos.create_and_set_qos:
-
args:
consumer: "back-end"
write_iops_sec: "10"
read_iops_sec: "1000"
set_consumer: "both"
set_write_iops_sec: "11"
set_read_iops_sec: "1001"
runner:
type: "constant"
times: 5
concurrency: 2
context:
users:
tenants: 2
users_per_tenant: 2
sla:
failure_rate:
max: 0

View File

@ -27,11 +27,19 @@ from rally.task import validation
@scenario.configure(context={"admin_cleanup": ["cinder"]}, @scenario.configure(context={"admin_cleanup": ["cinder"]},
name="CinderQos.create_and_list_qos") name="CinderQos.create_and_list_qos")
class CreateAndListQos(cinder_utils.CinderBasic): class CreateAndListQos(cinder_utils.CinderBasic):
def run(self, specs): def run(self, consumer, write_iops_sec, read_iops_sec):
"""create a qos, then list all qos. """Create a qos, then list all qos.
:param specs: A dict of key/value pairs to create qos :param consumer: Consumer behavior
:param write_iops_sec: random write limitation
:param read_iops_sec: random read limitation
""" """
specs = {
"consumer": consumer,
"write_iops_sec": write_iops_sec,
"read_iops_sec": read_iops_sec
}
qos = self.admin_cinder.create_qos(specs) qos = self.admin_cinder.create_qos(specs)
pool_list = self.admin_cinder.list_qos() pool_list = self.admin_cinder.list_qos()
@ -47,10 +55,49 @@ class CreateAndListQos(cinder_utils.CinderBasic):
@scenario.configure(context={"admin_cleanup": ["cinder"]}, @scenario.configure(context={"admin_cleanup": ["cinder"]},
name="CinderQos.create_and_get_qos") name="CinderQos.create_and_get_qos")
class CreateAndGetQos(cinder_utils.CinderBasic): class CreateAndGetQos(cinder_utils.CinderBasic):
def run(self, specs): def run(self, consumer, write_iops_sec, read_iops_sec):
"""Create a qos, then get details of the qos. """Create a qos, then get details of the qos.
:param specs: A dict of key/value pairs to create qos :param consumer: Consumer behavior
:param write_iops_sec: random write limitation
:param read_iops_sec: random read limitation
""" """
specs = {
"consumer": consumer,
"write_iops_sec": write_iops_sec,
"read_iops_sec": read_iops_sec
}
qos = self.admin_cinder.create_qos(specs) qos = self.admin_cinder.create_qos(specs)
self.admin_cinder.get_qos(qos.id) self.admin_cinder.get_qos(qos.id)
@validation.add("required_services", services=[consts.Service.CINDER])
@validation.add("required_platform", platform="openstack", admin=True)
@scenario.configure(context={"admin_cleanup": ["cinder"]},
name="CinderQos.create_and_set_qos")
class CreateAndSetQos(cinder_utils.CinderBasic):
def run(self, consumer, write_iops_sec, read_iops_sec,
set_consumer, set_write_iops_sec, set_read_iops_sec):
"""Create a qos, then Add/Update keys in qos specs.
:param consumer: Consumer behavior
:param write_iops_sec: random write limitation
:param read_iops_sec: random read limitation
:param set_consumer: update Consumer behavior
:param set_write_iops_sec: update random write limitation
:param set_read_iops_sec: update random read limitation
"""
create_specs = {
"consumer": consumer,
"write_iops_sec": write_iops_sec,
"read_iops_sec": read_iops_sec
}
set_specs = {
"consumer": set_consumer,
"write_iops_sec": set_write_iops_sec,
"read_iops_sec": set_read_iops_sec
}
qos = self.admin_cinder.create_qos(create_specs)
self.admin_cinder.set_qos(qos=qos, set_specs_args=set_specs)

View File

@ -34,7 +34,7 @@ VolumeTransfer = collections.namedtuple("VolumeTransfer", ["id", "name",
"auth_key"]) "auth_key"])
VolumeEncryptionType = collections.namedtuple("VolumeEncryptionType", VolumeEncryptionType = collections.namedtuple("VolumeEncryptionType",
["id", "volume_type_id"]) ["id", "volume_type_id"])
QoSSpecs = collections.namedtuple("QoSSpecs", ["id", "name"]) QoSSpecs = collections.namedtuple("QoSSpecs", ["id", "name", "specs"])
class BlockStorage(service.UnifiedService): class BlockStorage(service.UnifiedService):
@ -217,6 +217,17 @@ class BlockStorage(service.UnifiedService):
""" """
return self._impl.get_qos(qos_id) return self._impl.get_qos(qos_id)
@service.should_be_overridden
def set_qos(self, qos, set_specs_args):
"""Add/Update keys in qos specs.
:param qos: The instance of the :class:`QoSSpecs` to set
:param set_specs_args: A dict of key/value pairs to be set
:rtype: :class:`QoSSpecs`
"""
return self._impl.set_qos(qos=qos,
set_specs_args=set_specs_args)
@service.should_be_overridden @service.should_be_overridden
def create_snapshot(self, volume_id, force=False, def create_snapshot(self, volume_id, force=False,
name=None, description=None, metadata=None): name=None, description=None, metadata=None):

View File

@ -235,6 +235,19 @@ class CinderMixin(object):
with atomic.ActionTimer(self, aname): with atomic.ActionTimer(self, aname):
return self._get_client().qos_specs.get(qos_id) return self._get_client().qos_specs.get(qos_id)
def set_qos(self, qos_id, set_specs_args):
"""Add/Update keys in qos specs.
:param qos_id: The ID of the :class:`QoSSpecs` to get
:param set_specs_args: A dict of key/value pairs to be set
:rtype: class 'cinderclient.apiclient.base.DictWithMeta'
{"qos_specs": set_specs_args}
"""
aname = "cinder_v%s.set_qos" % self.version
with atomic.ActionTimer(self, aname):
return self._get_client().qos_specs.set_keys(qos_id,
set_specs_args)
def delete_snapshot(self, snapshot): def delete_snapshot(self, snapshot):
"""Delete the given snapshot. """Delete the given snapshot.
@ -444,7 +457,7 @@ class UnifiedCinderMixin(object):
@staticmethod @staticmethod
def _unify_qos(qos): def _unify_qos(qos):
return block.QoSSpecs(id=qos.id, name=qos.name) return block.QoSSpecs(id=qos.id, name=qos.name, specs=qos.specs)
@staticmethod @staticmethod
def _unify_encryption_type(encryption_type): def _unify_encryption_type(encryption_type):
@ -534,6 +547,16 @@ class UnifiedCinderMixin(object):
""" """
return self._unify_qos(self._impl.get_qos(qos_id)) return self._unify_qos(self._impl.get_qos(qos_id))
def set_qos(self, qos, set_specs_args):
"""Add/Update keys in qos specs.
:param qos: The instance of the :class:`QoSSpecs` to set
:param set_specs_args: A dict of key/value pairs to be set
:rtype: :class: 'QoSSpecs'
"""
self._impl.set_qos(qos.id, set_specs_args)
return self._unify_qos(qos)
def delete_snapshot(self, snapshot): def delete_snapshot(self, snapshot):
"""Delete the given backup. """Delete the given backup.

View File

@ -2,11 +2,9 @@
"CinderQos.create_and_get_qos": [ "CinderQos.create_and_get_qos": [
{ {
"args": { "args": {
"specs": { "consumer": "both",
"consumer": "both", "write_iops_sec": "10",
"write_iops_sec": "10", "read_iops_sec": "1000"
"read_iops_sec": "1000"
}
}, },
"runner": { "runner": {
"type": "constant", "type": "constant",

View File

@ -2,10 +2,9 @@
CinderQos.create_and_get_qos: CinderQos.create_and_get_qos:
- -
args: args:
specs: consumer: "both"
consumer: "both" write_iops_sec: "10"
write_iops_sec: "10" read_iops_sec: "1000"
read_iops_sec: "1000"
runner: runner:
type: "constant" type: "constant"
times: 5 times: 5

View File

@ -2,11 +2,9 @@
"CinderQos.create_and_list_qos": [ "CinderQos.create_and_list_qos": [
{ {
"args": { "args": {
"specs": { "consumer": "both",
"consumer": "both", "write_iops_sec": "10",
"write_iops_sec": "10", "read_iops_sec": "1000"
"read_iops_sec": "1000"
}
}, },
"runner": { "runner": {
"type": "constant", "type": "constant",

View File

@ -2,10 +2,9 @@
CinderQos.create_and_list_qos: CinderQos.create_and_list_qos:
- -
args: args:
specs: consumer: "both"
consumer: "both" write_iops_sec: "10"
write_iops_sec: "10" read_iops_sec: "1000"
read_iops_sec: "1000"
runner: runner:
type: "constant" type: "constant"
times: 5 times: 5

View File

@ -0,0 +1,30 @@
{
"CinderQos.create_and_set_qos": [
{
"args": {
"consumer": "back-end",
"write_iops_sec": "10",
"read_iops_sec": "1000",
"set_consumer": "both",
"set_write_iops_sec": "11",
"set_read_iops_sec": "1001"
},
"runner": {
"type": "constant",
"times": 5,
"concurrency": 2
},
"context": {
"users": {
"tenants": 2,
"users_per_tenant": 2
}
},
"sla": {
"failure_rate": {
"max": 0
}
}
}
]
}

View File

@ -0,0 +1,21 @@
---
CinderQos.create_and_set_qos:
-
args:
consumer: "back-end"
write_iops_sec: "10"
read_iops_sec: "1000"
set_consumer: "both"
set_write_iops_sec: "11"
set_read_iops_sec: "1001"
runner:
type: "constant"
times: 5
concurrency: 2
context:
users:
tenants: 2
users_per_tenant: 2
sla:
failure_rate:
max: 0

View File

@ -55,7 +55,7 @@ class CinderQosTestCase(test.ScenarioTestCase):
mock_service.create_qos.return_value = qos mock_service.create_qos.return_value = qos
mock_service.list_qos.return_value = list_qos mock_service.list_qos.return_value = list_qos
scenario.run(specs) scenario.run("both", "10", "1000")
mock_service.create_qos.assert_called_once_with(specs) mock_service.create_qos.assert_called_once_with(specs)
mock_service.list_qos.assert_called_once_with() mock_service.list_qos.assert_called_once_with()
@ -74,7 +74,7 @@ class CinderQosTestCase(test.ScenarioTestCase):
mock_service.list_qos.return_value = list_qos mock_service.list_qos.return_value = list_qos
self.assertRaises(rally_exceptions.RallyAssertionError, self.assertRaises(rally_exceptions.RallyAssertionError,
scenario.run, specs) scenario.run, "both", "10", "1000")
mock_service.create_qos.assert_called_once_with(specs) mock_service.create_qos.assert_called_once_with(specs)
mock_service.list_qos.assert_called_once_with() mock_service.list_qos.assert_called_once_with()
@ -88,6 +88,25 @@ class CinderQosTestCase(test.ScenarioTestCase):
scenario = qos_specs.CreateAndGetQos(self._get_context()) scenario = qos_specs.CreateAndGetQos(self._get_context())
mock_service.create_qos.return_value = qos mock_service.create_qos.return_value = qos
scenario.run(specs) scenario.run("both", "10", "1000")
mock_service.create_qos.assert_called_once_with(specs) mock_service.create_qos.assert_called_once_with(specs)
mock_service.get_qos.assert_called_once_with(qos.id) mock_service.get_qos.assert_called_once_with(qos.id)
def test_create_and_set_qos(self):
mock_service = self.mock_cinder.return_value
qos = mock.MagicMock()
create_specs_args = {"consumer": "back-end",
"write_iops_sec": "10",
"read_iops_sec": "1000"}
set_specs_args = {"consumer": "both",
"write_iops_sec": "11",
"read_iops_sec": "1001"}
scenario = qos_specs.CreateAndSetQos(self._get_context())
mock_service.create_qos.return_value = qos
scenario.run("back-end", "10", "1000",
"both", "11", "1001")
mock_service.create_qos.assert_called_once_with(create_specs_args)
mock_service.set_qos.assert_called_once_with(
qos=qos, set_specs_args=set_specs_args)

View File

@ -137,6 +137,14 @@ class BlockTestCase(test.TestCase):
self.service.get_qos("qos")) self.service.get_qos("qos"))
self.service._impl.get_qos.assert_called_once_with("qos") self.service._impl.get_qos.assert_called_once_with("qos")
def test_set_qos(self):
set_specs_args = {"test": "foo"}
self.assertEqual(
self.service._impl.set_qos.return_value,
self.service.set_qos(qos="qos", set_specs_args=set_specs_args))
self.service._impl.set_qos.assert_called_once_with(
qos="qos", set_specs_args=set_specs_args)
def test_create_snapshot(self): def test_create_snapshot(self):
self.assertEqual( self.assertEqual(
self.service._impl.create_snapshot.return_value, self.service._impl.create_snapshot.return_value,

View File

@ -258,6 +258,15 @@ class CinderMixinTestCase(test.ScenarioTestCase):
result) result)
self.cinder.qos_specs.get.assert_called_once_with("qos") self.cinder.qos_specs.get.assert_called_once_with("qos")
def test_set_qos(self):
set_specs_args = {"test": "foo"}
result = self.service.set_qos("qos", set_specs_args)
self.assertEqual(
self.cinder.qos_specs.set_keys.return_value,
result)
self.cinder.qos_specs.set_keys.assert_called_once_with("qos",
set_specs_args)
def test_delete_snapshot(self): def test_delete_snapshot(self):
snapshot = mock.Mock() snapshot = mock.Mock()
self.service.delete_snapshot(snapshot) self.service.delete_snapshot(snapshot)
@ -466,10 +475,12 @@ class UnifiedCinderMixinTestCase(test.TestCase):
def test__unify_qos(self): def test__unify_qos(self):
class Qos(object): class Qos(object):
id = 1 id = 1
name = "transfer" name = "qos"
specs = {"key1": "value1"}
qos = self.service._unify_qos(Qos()) qos = self.service._unify_qos(Qos())
self.assertEqual(1, qos.id) self.assertEqual(1, qos.id)
self.assertEqual("transfer", qos.name) self.assertEqual("qos", qos.name)
self.assertEqual({"key1": "value1"}, qos.specs)
def test__unify_encryption_type(self): def test__unify_encryption_type(self):
class SomeEncryptionType(object): class SomeEncryptionType(object):
@ -549,6 +560,17 @@ class UnifiedCinderMixinTestCase(test.TestCase):
self.service._impl.get_qos.return_value self.service._impl.get_qos.return_value
) )
def test_set_qos(self):
set_specs_args = {"test": "foo"}
self.service._unify_qos = mock.MagicMock()
qos = mock.MagicMock()
self.assertEqual(
self.service._unify_qos.return_value,
self.service.set_qos(qos, set_specs_args))
self.service._impl.set_qos.assert_called_once_with(qos.id,
set_specs_args)
self.service._unify_qos.assert_called_once_with(qos)
def test_delete_snapshot(self): def test_delete_snapshot(self):
self.service.delete_snapshot("snapshot") self.service.delete_snapshot("snapshot")
self.service._impl.delete_snapshot.assert_called_once_with("snapshot") self.service._impl.delete_snapshot.assert_called_once_with("snapshot")