NetApp Solidfire: Fix retype and name exception on migration
While performing retype or migrate on a volume the name exception occurs due to the check on all backends including the origin. The retype status exception occurs because the only accepted state is available, but retype changes the status to retyping. This change just assures to check the name on the target backend and also adds retyping as a valid status for migrate condition. Change-Id: Ic09d65a0a6ecbb529dc0e4b041650ead449e1273 Closes-Bug: 1942090 Closes-Bug: 1932964
This commit is contained in:
parent
a8da1fa83b
commit
853f082884
@ -3728,6 +3728,32 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
sfv.migrate_volume, ctx, vol, host)
|
sfv.migrate_volume, ctx, vol, host)
|
||||||
mock_do_intercluster_volume_migration.assert_not_called()
|
mock_do_intercluster_volume_migration.assert_not_called()
|
||||||
|
|
||||||
|
@mock.patch.object(solidfire.SolidFireDriver,
|
||||||
|
'_do_intercluster_volume_migration')
|
||||||
|
def test_migrate_volume_retyping_status(
|
||||||
|
self, mock_do_intercluster_volume_migration):
|
||||||
|
|
||||||
|
ctx = context.get_admin_context()
|
||||||
|
type_fields = {'id': fakes.get_fake_uuid()}
|
||||||
|
src_vol_type = fake_volume.fake_volume_type_obj(ctx, **type_fields)
|
||||||
|
|
||||||
|
vol_fields = {
|
||||||
|
'id': fakes.get_fake_uuid(),
|
||||||
|
'volume_type': src_vol_type,
|
||||||
|
'host': 'fakeHost@fakeBackend#fakePool',
|
||||||
|
'status': 'retyping'
|
||||||
|
}
|
||||||
|
|
||||||
|
vol = fake_volume.fake_volume_obj(ctx, **vol_fields)
|
||||||
|
vol.volume_type = src_vol_type
|
||||||
|
host = {'host': 'fakeHost@fakeBackend#fakePool'}
|
||||||
|
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
result = sfv.migrate_volume(ctx, vol, host)
|
||||||
|
|
||||||
|
mock_do_intercluster_volume_migration.assert_not_called()
|
||||||
|
self.assertEqual((True, {}), result)
|
||||||
|
|
||||||
@mock.patch.object(solidfire.SolidFireDriver,
|
@mock.patch.object(solidfire.SolidFireDriver,
|
||||||
'_do_intercluster_volume_migration')
|
'_do_intercluster_volume_migration')
|
||||||
def test_migrate_volume_same_host_and_backend(
|
def test_migrate_volume_same_host_and_backend(
|
||||||
@ -4192,3 +4218,40 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
call('RemoveVolumePair', src_params, '8.0'),
|
call('RemoveVolumePair', src_params, '8.0'),
|
||||||
call('DeleteVolume', src_params),
|
call('DeleteVolume', src_params),
|
||||||
call('PurgeDeletedVolume', src_params)])
|
call('PurgeDeletedVolume', src_params)])
|
||||||
|
|
||||||
|
@data(True, False)
|
||||||
|
@mock.patch.object(solidfire.SolidFireDriver, '_create_cluster_reference')
|
||||||
|
@mock.patch.object(solidfire.SolidFireDriver, '_set_cluster_pairs')
|
||||||
|
@mock.patch.object(solidfire.SolidFireDriver, '_update_cluster_status')
|
||||||
|
@mock.patch.object(solidfire.SolidFireDriver, '_issue_api_request')
|
||||||
|
def test_list_volumes_by_name(self, has_endpoint, mock_issue_api_request,
|
||||||
|
mock_update_cluster_status,
|
||||||
|
mock_set_cluster_pairs,
|
||||||
|
mock_create_cluster_reference):
|
||||||
|
fake_sf_volume_name = 'fake-vol-name'
|
||||||
|
vol_fields = {
|
||||||
|
'id': fakes.get_fake_uuid(),
|
||||||
|
'name': fake_sf_volume_name
|
||||||
|
}
|
||||||
|
vol = fake_volume.fake_volume_obj(
|
||||||
|
context.get_admin_context(), **vol_fields)
|
||||||
|
|
||||||
|
fake_endpoint = None
|
||||||
|
volumes_list = [vol]
|
||||||
|
|
||||||
|
if has_endpoint:
|
||||||
|
fake_endpoint = self.fake_primary_cluster["endpoint"]
|
||||||
|
volumes_list = []
|
||||||
|
|
||||||
|
mock_issue_api_request.return_value = {
|
||||||
|
'result': {'volumes': volumes_list}}
|
||||||
|
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
result = (
|
||||||
|
sfv._list_volumes_by_name(fake_sf_volume_name,
|
||||||
|
endpoint=fake_endpoint))
|
||||||
|
|
||||||
|
mock_issue_api_request.assert_called_once_with(
|
||||||
|
'ListVolumes', {'volumeName': fake_sf_volume_name},
|
||||||
|
version='8.0', endpoint=fake_endpoint)
|
||||||
|
self.assertEqual(result, volumes_list)
|
||||||
|
@ -276,9 +276,13 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
by adding xNotPrimary to the retryable exception list
|
by adding xNotPrimary to the retryable exception list
|
||||||
2.2.2 - Fix bug #1896112 SolidFire Driver creates duplicate volume
|
2.2.2 - Fix bug #1896112 SolidFire Driver creates duplicate volume
|
||||||
when API response is lost
|
when API response is lost
|
||||||
|
2.2.3 - Fix bug #1942090 SolidFire retype fails due to volume status
|
||||||
|
as retyping.
|
||||||
|
Fix bug #1932964 SolidFire duplicate volume name exception
|
||||||
|
on migration and replication.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
VERSION = '2.2.2'
|
VERSION = '2.2.3'
|
||||||
|
|
||||||
SUPPORTS_ACTIVE_ACTIVE = True
|
SUPPORTS_ACTIVE_ACTIVE = True
|
||||||
|
|
||||||
@ -1015,10 +1019,10 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
params['attributes'] = attributes
|
params['attributes'] = attributes
|
||||||
return self._issue_api_request('ModifyVolume', params)
|
return self._issue_api_request('ModifyVolume', params)
|
||||||
|
|
||||||
def _list_volumes_by_name(self, sf_volume_name):
|
def _list_volumes_by_name(self, sf_volume_name, endpoint=None):
|
||||||
params = {'volumeName': sf_volume_name}
|
params = {'volumeName': sf_volume_name}
|
||||||
return self._issue_api_request(
|
return self._issue_api_request('ListVolumes', params, version='8.0',
|
||||||
'ListVolumes', params, version='8.0')['result']['volumes']
|
endpoint=endpoint)['result']['volumes']
|
||||||
|
|
||||||
def _wait_volume_is_active(self, sf_volume_name):
|
def _wait_volume_is_active(self, sf_volume_name):
|
||||||
|
|
||||||
@ -1046,9 +1050,10 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
raise SolidFireAPIException(msg)
|
raise SolidFireAPIException(msg)
|
||||||
|
|
||||||
def _do_volume_create(self, sf_account, params, endpoint=None):
|
def _do_volume_create(self, sf_account, params, endpoint=None):
|
||||||
|
|
||||||
sf_volume_name = params['name']
|
sf_volume_name = params['name']
|
||||||
volumes_found = self._list_volumes_by_name(sf_volume_name)
|
volumes_found = self._list_volumes_by_name(sf_volume_name,
|
||||||
|
endpoint=endpoint)
|
||||||
|
|
||||||
if volumes_found:
|
if volumes_found:
|
||||||
raise SolidFireDuplicateVolumeNames(vol_name=sf_volume_name)
|
raise SolidFireDuplicateVolumeNames(vol_name=sf_volume_name)
|
||||||
|
|
||||||
@ -2522,9 +2527,10 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
LOG.info("Migrate volume %(vol_id)s to %(host)s.",
|
LOG.info("Migrate volume %(vol_id)s to %(host)s.",
|
||||||
{"vol_id": volume.id, "host": host["host"]})
|
{"vol_id": volume.id, "host": host["host"]})
|
||||||
|
|
||||||
if volume.status != fields.VolumeStatus.AVAILABLE:
|
if (volume.status != fields.VolumeStatus.AVAILABLE and
|
||||||
msg = _("Volume status must be 'available' to execute "
|
volume.status != fields.VolumeStatus.RETYPING):
|
||||||
"storage assisted migration.")
|
msg = _("Volume status must be 'available' or 'retyping' to "
|
||||||
|
"execute storage assisted migration.")
|
||||||
LOG.error(msg)
|
LOG.error(msg)
|
||||||
raise exception.InvalidVolume(reason=msg)
|
raise exception.InvalidVolume(reason=msg)
|
||||||
|
|
||||||
|
@ -0,0 +1,10 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
NetApp SolidFire driver `Bug #1932964
|
||||||
|
<https://bugs.launchpad.net/cinder/+bug/1932964>`_:
|
||||||
|
Fixed a name exception that occurs on any volume migration.
|
||||||
|
- |
|
||||||
|
NetApp SolidFire driver `Bug #1942090
|
||||||
|
<https://bugs.launchpad.net/cinder/+bug/1942090>`_:
|
||||||
|
Fixed a status exception that occurs on volume retype with migration.
|
Loading…
Reference in New Issue
Block a user