Add possibility to create lots of shares in parallel for tempest

Test suite located in module
contrib/tempest/tempest/api/share/admin/test_multi_backend.py
creates 2 shares without waiting for "available" status and then waits for
status. In this case "retry" functionality is not used.

Add method 'create_shares' to base class that does parallel share
creation with recreation as it is done with single request. And use this
method in mentioned test module.

Change-Id: I1314c9494a547314cd9bb9b03323470d58956fac
Closes-Bug: #1406770
This commit is contained in:
Valeriy Ponomaryov 2014-12-31 17:06:24 +02:00 committed by vponomaryov
parent 0dd28ab401
commit a52d1b49a6
2 changed files with 81 additions and 35 deletions

View File

@ -36,6 +36,7 @@ class ShareMultiBackendTest(base.BaseSharesAdminTest):
"Skipping.")
cls.vts = []
cls.shares = []
share_data_list = []
# Create volume types
for i in [0, 1]:
@ -46,16 +47,10 @@ class ShareMultiBackendTest(base.BaseSharesAdminTest):
__, vt = cls.create_volume_type(name=vt_name,
extra_specs=extra_specs)
cls.vts.append(vt)
share_data_list.append({"kwargs": {"volume_type_id": vt["id"]}})
# Create shares
for vt in cls.vts:
__, share = cls.create_share(volume_type_id=vt["id"],
wait_for_active=False)
cls.shares.append(share)
# Wait for active statuses of shares
for share in cls.shares:
cls.shares_client.wait_for_share_status(share["id"], "available")
# Create shares using precreated volume types
cls.shares = cls.create_shares(share_data_list)
@test.attr(type=["gate", "smoke", ])
def test_share_backend_name_reporting(self):

View File

@ -13,6 +13,7 @@
# License for the specific language governing permissions and limitations
# under the License.
import copy
import inspect
from tempest import clients_share as clients
@ -227,15 +228,15 @@ class BaseSharesTest(test.BaseTestCase):
return share_network_id
@classmethod
def create_share(cls, share_protocol=None, size=1, name=None,
snapshot_id=None, description=None, metadata={},
def _create_share(cls, share_protocol=None, size=1, name=None,
snapshot_id=None, description=None, metadata=None,
share_network_id=None, volume_type_id=None,
client=None, cleanup_in_class=True,
wait_for_active=True):
client=None, cleanup_in_class=True):
client = client or cls.shares_client
description = description or "Tempest's share"
share_network_id = share_network_id or client.share_network_id or None
kw = {
metadata = metadata or {}
kwargs = {
'share_protocol': share_protocol,
'size': size,
'name': name,
@ -245,8 +246,6 @@ class BaseSharesTest(test.BaseTestCase):
'share_network_id': share_network_id,
'volume_type_id': volume_type_id,
}
def _create_share(**kwargs):
resp, share = client.create_share(**kwargs)
resource = {"type": "share", "id": share["id"], "client": client}
cleanup_list = (cls.class_resources if cleanup_in_class else
@ -254,24 +253,76 @@ class BaseSharesTest(test.BaseTestCase):
cleanup_list.insert(0, resource)
return resp, share
resp, share = _create_share(**kw)
if wait_for_active:
cnt = 0
while True:
if cnt > 0:
resp, share = _create_share(**kw)
@classmethod
def create_share(cls, *args, **kwargs):
"""Create one share and wait for available state. Retry if allowed."""
result = cls.create_shares(
[{"args": args, "kwargs": kwargs}], with_resp=True)
return result[0]
@classmethod
def create_shares(cls, share_data_list, with_resp=False):
"""Creates several shares in parallel with retries.
Use this method when you want to create more than one share at same
time. Especially if config option 'share.share_creation_retry_number'
has value more than zero (0).
All shares will be expected to have 'available' status with or without
recreation else error will be raised.
:param share_data_list: list -- list of dictionaries with 'args' and
'kwargs' for '_create_share' method of this base class.
example of data:
share_data_list=[{'args': ['quuz'], 'kwargs': {'foo': 'bar'}}}]
:param with_resp: boolean -- whether to return list of
tuples (resp, share) or just list of shares.
:returns: list -- list of shares created using provided data.
"""
data = [copy.deepcopy(d) for d in share_data_list]
for d in data:
if not isinstance(d, dict):
raise exceptions.TempestException(
"Expected 'dict', got '%s'" % type(d))
if "args" not in d:
d["args"] = []
if "kwargs" not in d:
d["kwargs"] = {}
if len(d) > 2:
raise exceptions.TempestException(
"Expected only 'args' and 'kwargs' keys. "
"Provided %s" % list(d))
d["kwargs"]["client"] = d["kwargs"].get(
"client", cls.shares_client)
d["resp"], d["share"] = cls._create_share(
*d["args"], **d["kwargs"])
d["cnt"] = 0
d["available"] = False
while not all(d["available"] for d in data):
for d in data:
if d["available"]:
continue
try:
client.wait_for_share_status(share["id"], "available")
d["kwargs"]["client"].wait_for_share_status(
d["share"]["id"], "available")
d["available"] = True
except (share_exceptions.ShareBuildErrorException,
exceptions.TimeoutException) as e:
if CONF.share.share_creation_retry_number > cnt:
cnt += 1
continue
if CONF.share.share_creation_retry_number > d["cnt"]:
d["cnt"] += 1
msg = ("Share '%(id)s' failed to be built. "
"Trying create another." % d["share"]["id"])
LOG.error(msg)
LOG.error(e)
d["resp"], d["share"] = cls._create_share(
*d["args"], **d["kwargs"])
else:
raise e
else:
break
return resp, share
if with_resp:
return [(d["resp"], d["share"]) for d in data]
return [d["share"] for d in data]
@classmethod
def create_snapshot_wait_for_active(cls, share_id, name=None,