Merge "Refactor 'update_group' method"
This commit is contained in:
commit
e726321d97
@ -140,6 +140,53 @@ class GroupManagerTestCase(test.TestCase):
|
|||||||
self.context,
|
self.context,
|
||||||
group.id)
|
group.id)
|
||||||
|
|
||||||
|
@ddt.data(('', [], 0, None, True),
|
||||||
|
('1,2', ['available', 'in-use'], 2, None, True),
|
||||||
|
('1,2,3', ['available', 'in-use', 'error_deleting'], 3,
|
||||||
|
None, False),
|
||||||
|
('1,2', ['wrong_status', 'available'], 0,
|
||||||
|
exception.InvalidVolume, True),
|
||||||
|
('1,2', ['available', exception.VolumeNotFound],
|
||||||
|
0, exception.VolumeNotFound, True))
|
||||||
|
@ddt.unpack
|
||||||
|
@mock.patch('cinder.objects.Volume.get_by_id')
|
||||||
|
def test__collect_volumes_for_group(self, add_volumes, returned, expected,
|
||||||
|
raise_error, add, mock_get):
|
||||||
|
side_effect = []
|
||||||
|
|
||||||
|
class FakeVolume(object):
|
||||||
|
def __init__(self, status):
|
||||||
|
self.status = status
|
||||||
|
self.id = fake.UUID1
|
||||||
|
|
||||||
|
for value in returned:
|
||||||
|
if isinstance(value, str):
|
||||||
|
value = FakeVolume(value)
|
||||||
|
else:
|
||||||
|
value = value(volume_id=fake.UUID1)
|
||||||
|
side_effect.append(value)
|
||||||
|
mock_get.side_effect = side_effect
|
||||||
|
group = tests_utils.create_group(
|
||||||
|
self.context,
|
||||||
|
availability_zone=CONF.storage_availability_zone,
|
||||||
|
volume_type_ids=[fake.VOLUME_TYPE_ID],
|
||||||
|
group_type_id=fake.GROUP_TYPE_ID,
|
||||||
|
host=CONF.host)
|
||||||
|
|
||||||
|
with mock.patch.object(self.volume, '_check_is_our_resource',
|
||||||
|
mock.Mock()) as mock_check:
|
||||||
|
if raise_error:
|
||||||
|
self.assertRaises(raise_error,
|
||||||
|
self.volume._collect_volumes_for_group,
|
||||||
|
None, group, add_volumes, add)
|
||||||
|
else:
|
||||||
|
result = self.volume._collect_volumes_for_group(None, group,
|
||||||
|
add_volumes,
|
||||||
|
add=add)
|
||||||
|
if add:
|
||||||
|
self.assertEqual(expected, mock_check.call_count)
|
||||||
|
self.assertEqual(expected, len(result))
|
||||||
|
|
||||||
@ddt.data((False, fake.GROUP_TYPE_ID),
|
@ddt.data((False, fake.GROUP_TYPE_ID),
|
||||||
(True, fake.GROUP_TYPE_ID),
|
(True, fake.GROUP_TYPE_ID),
|
||||||
(True, fake.GROUP_TYPE2_ID))
|
(True, fake.GROUP_TYPE2_ID))
|
||||||
|
@ -3097,6 +3097,41 @@ class VolumeManager(manager.CleanableManager,
|
|||||||
# anything in the backend storage.
|
# anything in the backend storage.
|
||||||
return None, None, None
|
return None, None, None
|
||||||
|
|
||||||
|
def _collect_volumes_for_group(self, context, group, volumes, add=True):
|
||||||
|
if add:
|
||||||
|
valid_status = VALID_ADD_VOL_TO_GROUP_STATUS
|
||||||
|
else:
|
||||||
|
valid_status = VALID_REMOVE_VOL_FROM_GROUP_STATUS
|
||||||
|
volumes_ref = []
|
||||||
|
if not volumes:
|
||||||
|
return volumes_ref
|
||||||
|
for add_vol in volumes.split(','):
|
||||||
|
try:
|
||||||
|
add_vol_ref = objects.Volume.get_by_id(context, add_vol)
|
||||||
|
except exception.VolumeNotFound:
|
||||||
|
LOG.error("Update group "
|
||||||
|
"failed to %(op)s volume-%(volume_id)s: "
|
||||||
|
"VolumeNotFound.",
|
||||||
|
{'volume_id': add_vol_ref.id,
|
||||||
|
'op': 'add' if add else 'remove'},
|
||||||
|
resource={'type': 'group',
|
||||||
|
'id': group.id})
|
||||||
|
raise
|
||||||
|
if add_vol_ref.status not in valid_status:
|
||||||
|
msg = (_("Can not %(op)s volume %(volume_id)s to "
|
||||||
|
"group %(group_id)s because volume is in an invalid "
|
||||||
|
"state: %(status)s. Valid states are: %(valid)s.") %
|
||||||
|
{'volume_id': add_vol_ref.id,
|
||||||
|
'group_id': group.id,
|
||||||
|
'status': add_vol_ref.status,
|
||||||
|
'valid': valid_status,
|
||||||
|
'op': 'add' if add else 'remove'})
|
||||||
|
raise exception.InvalidVolume(reason=msg)
|
||||||
|
if add:
|
||||||
|
self._check_is_our_resource(add_vol_ref)
|
||||||
|
volumes_ref.append(add_vol_ref)
|
||||||
|
return volumes_ref
|
||||||
|
|
||||||
def update_group(self, context, group,
|
def update_group(self, context, group,
|
||||||
add_volumes=None, remove_volumes=None):
|
add_volumes=None, remove_volumes=None):
|
||||||
"""Updates group.
|
"""Updates group.
|
||||||
@ -3105,60 +3140,14 @@ class VolumeManager(manager.CleanableManager,
|
|||||||
or removing volumes from the group.
|
or removing volumes from the group.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
add_volumes_ref = []
|
add_volumes_ref = self._collect_volumes_for_group(context,
|
||||||
remove_volumes_ref = []
|
group,
|
||||||
add_volumes_list = []
|
add_volumes,
|
||||||
remove_volumes_list = []
|
add=True)
|
||||||
if add_volumes:
|
remove_volumes_ref = self._collect_volumes_for_group(context,
|
||||||
add_volumes_list = add_volumes.split(',')
|
group,
|
||||||
if remove_volumes:
|
remove_volumes,
|
||||||
remove_volumes_list = remove_volumes.split(',')
|
add=False)
|
||||||
for add_vol in add_volumes_list:
|
|
||||||
try:
|
|
||||||
add_vol_ref = objects.Volume.get_by_id(context, add_vol)
|
|
||||||
except exception.VolumeNotFound:
|
|
||||||
LOG.error("Update group "
|
|
||||||
"failed to add volume-%(volume_id)s: "
|
|
||||||
"VolumeNotFound.",
|
|
||||||
{'volume_id': add_vol_ref.id},
|
|
||||||
resource={'type': 'group',
|
|
||||||
'id': group.id})
|
|
||||||
raise
|
|
||||||
if add_vol_ref.status not in VALID_ADD_VOL_TO_GROUP_STATUS:
|
|
||||||
msg = (_("Cannot add volume %(volume_id)s to "
|
|
||||||
"group %(group_id)s because volume is in an invalid "
|
|
||||||
"state: %(status)s. Valid states are: %(valid)s.") %
|
|
||||||
{'volume_id': add_vol_ref.id,
|
|
||||||
'group_id': group.id,
|
|
||||||
'status': add_vol_ref.status,
|
|
||||||
'valid': VALID_ADD_VOL_TO_GROUP_STATUS})
|
|
||||||
raise exception.InvalidVolume(reason=msg)
|
|
||||||
self._check_is_our_resource(add_vol_ref)
|
|
||||||
add_volumes_ref.append(add_vol_ref)
|
|
||||||
|
|
||||||
for remove_vol in remove_volumes_list:
|
|
||||||
try:
|
|
||||||
remove_vol_ref = objects.Volume.get_by_id(context, remove_vol)
|
|
||||||
except exception.VolumeNotFound:
|
|
||||||
LOG.error("Update group "
|
|
||||||
"failed to remove volume-%(volume_id)s: "
|
|
||||||
"VolumeNotFound.",
|
|
||||||
{'volume_id': remove_vol_ref.id},
|
|
||||||
resource={'type': 'group',
|
|
||||||
'id': group.id})
|
|
||||||
raise
|
|
||||||
if (remove_vol_ref.status not in
|
|
||||||
VALID_REMOVE_VOL_FROM_GROUP_STATUS):
|
|
||||||
msg = (_("Cannot remove volume %(volume_id)s from "
|
|
||||||
"group %(group_id)s because volume is in an invalid "
|
|
||||||
"state: %(status)s. Valid states are: %(valid)s.") %
|
|
||||||
{'volume_id': remove_vol_ref.id,
|
|
||||||
'group_id': group.id,
|
|
||||||
'status': remove_vol_ref.status,
|
|
||||||
'valid': VALID_REMOVE_VOL_FROM_GROUP_STATUS})
|
|
||||||
raise exception.InvalidVolume(reason=msg)
|
|
||||||
remove_volumes_ref.append(remove_vol_ref)
|
|
||||||
|
|
||||||
self._notify_about_group_usage(
|
self._notify_about_group_usage(
|
||||||
context, group, "update.start")
|
context, group, "update.start")
|
||||||
|
|
||||||
@ -3190,11 +3179,12 @@ class VolumeManager(manager.CleanableManager,
|
|||||||
self._remove_consistencygroup_id_from_volumes(
|
self._remove_consistencygroup_id_from_volumes(
|
||||||
remove_volumes_ref)
|
remove_volumes_ref)
|
||||||
|
|
||||||
|
volumes_to_update = []
|
||||||
if add_volumes_update:
|
if add_volumes_update:
|
||||||
self.db.volumes_update(context, add_volumes_update)
|
volumes_to_update.extend(add_volumes_update)
|
||||||
|
|
||||||
if remove_volumes_update:
|
if remove_volumes_update:
|
||||||
self.db.volumes_update(context, remove_volumes_update)
|
volumes_to_update.extend(remove_volumes_update)
|
||||||
|
self.db.volumes_update(context, volumes_to_update)
|
||||||
|
|
||||||
if model_update:
|
if model_update:
|
||||||
if model_update['status'] in (
|
if model_update['status'] in (
|
||||||
@ -3206,31 +3196,24 @@ class VolumeManager(manager.CleanableManager,
|
|||||||
group.update(model_update)
|
group.update(model_update)
|
||||||
group.save()
|
group.save()
|
||||||
|
|
||||||
except exception.VolumeDriverException:
|
except Exception as e:
|
||||||
with excutils.save_and_reraise_exception():
|
with excutils.save_and_reraise_exception():
|
||||||
LOG.error("Error occurred in the volume driver when "
|
if isinstance(e, exception.VolumeDriverException):
|
||||||
"updating group %(group_id)s.",
|
LOG.error("Error occurred in the volume driver when "
|
||||||
{'group_id': group.id})
|
"updating group %(group_id)s.",
|
||||||
group.status = 'error'
|
{'group_id': group.id})
|
||||||
group.save()
|
else:
|
||||||
for add_vol in add_volumes_ref:
|
LOG.error("Failed to update group %(group_id)s.",
|
||||||
add_vol.status = 'error'
|
{'group_id': group.id})
|
||||||
add_vol.save()
|
|
||||||
self._remove_consistencygroup_id_from_volumes(
|
|
||||||
remove_volumes_ref)
|
|
||||||
for rem_vol in remove_volumes_ref:
|
|
||||||
rem_vol.status = 'error'
|
|
||||||
rem_vol.save()
|
|
||||||
except Exception:
|
|
||||||
with excutils.save_and_reraise_exception():
|
|
||||||
LOG.error("Error occurred when updating group %(group_id)s.",
|
|
||||||
{'group_id': group.id})
|
|
||||||
group.status = 'error'
|
group.status = 'error'
|
||||||
group.save()
|
group.save()
|
||||||
for add_vol in add_volumes_ref:
|
for add_vol in add_volumes_ref:
|
||||||
add_vol.status = 'error'
|
add_vol.status = 'error'
|
||||||
add_vol.save()
|
add_vol.save()
|
||||||
for rem_vol in remove_volumes_ref:
|
for rem_vol in remove_volumes_ref:
|
||||||
|
if isinstance(e, exception.VolumeDriverException):
|
||||||
|
rem_vol.consistencygroup_id = None
|
||||||
|
rem_vol.consistencygroup = None
|
||||||
rem_vol.status = 'error'
|
rem_vol.status = 'error'
|
||||||
rem_vol.save()
|
rem_vol.save()
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user