[SVF]:Fix add volumes to GMCV group

[Spectrum Virtualize Family] Adding volumes to a GMCV group
fails even if the rccg and rcrelation are in same state.

This patch fixes the issues by keeping both rccg and rcrelation
in consistent_stopped state before updating the relation.

Also enabled snapshot support for GMCV volumes.

Closes-Bug: #1922013
Change-Id: Ic0bce20dd2b4ec607d278f78b55094110e2f72f6
This commit is contained in:
katarimanojkumar 2021-08-20 06:55:30 +00:00
parent f08af15a58
commit f882b0c657
3 changed files with 79 additions and 72 deletions

View File

@ -10235,24 +10235,33 @@ class StorwizeSVCReplicationTestCase(test.TestCase):
rel_info = self.driver._helpers.get_relationship_info(volume['name']) rel_info = self.driver._helpers.get_relationship_info(volume['name'])
self.assertIsNone(rel_info) self.assertIsNone(rel_info)
def test_storwize_create_snapshot_volume_with_mirror_replica(self): @ddt.data(({'mirror_type': 'mm_type'}),
({'mirror_type': 'gm_type'}),
({'mirror_type': 'gmcv_default_type'}),
({'mirror_type': 'gmcv_with_cps900_type'}))
def test_storwize_create_snapshot_volume_with_mirror_replica(self,
vol_spec):
# Set replication target # Set replication target
self.driver.configuration.set_override('replication_device', self.driver.configuration.set_override('replication_device',
[self.rep_target]) [self.rep_target])
self.driver.do_setup(self.ctxt) self.driver.do_setup(self.ctxt)
rep_type = getattr(self, vol_spec['mirror_type'])
# Create metro mirror replication volume. # Create metro mirror replication volume.
vol1, model_update = self._create_test_volume(self.mm_type) vol1, model_update = self._create_test_volume(rep_type)
self.assertEqual(fields.ReplicationStatus.ENABLED, self.assertEqual(fields.ReplicationStatus.ENABLED,
model_update['replication_status']) model_update['replication_status'])
snap = testutils.create_snapshot(self.ctxt, vol1.id) snap = testutils.create_snapshot(self.ctxt, vol1.id)
self.driver.create_snapshot(snap) self.driver.create_snapshot(snap)
self._assert_vol_exists(snap['name'], True)
vol2 = self._generate_vol_info(self.mm_type) vol2 = self._generate_vol_info(rep_type)
model_update = self.driver.create_volume_from_snapshot(vol2, snap) model_update = self.driver.create_volume_from_snapshot(vol2, snap)
self.assertEqual(fields.ReplicationStatus.ENABLED, self.assertEqual(fields.ReplicationStatus.ENABLED,
model_update['replication_status']) model_update['replication_status'])
if "gmcv" in vol_spec['mirror_type']:
self._validate_replic_vol_creation(vol2, isGMCV=True)
else:
self._validate_replic_vol_creation(vol2) self._validate_replic_vol_creation(vol2)
if self.USESIM: if self.USESIM:
@ -10261,28 +10270,6 @@ class StorwizeSVCReplicationTestCase(test.TestCase):
self.driver.delete_snapshot(snap) self.driver.delete_snapshot(snap)
self.driver.delete_volume(vol1) self.driver.delete_volume(vol1)
# Create gmcv replication volume.
vol1, model_update = self._create_test_volume(self.gmcv_default_type)
self.assertEqual(fields.ReplicationStatus.ENABLED,
model_update['replication_status'])
self._validate_replic_vol_creation(vol1, True)
snap = testutils.create_snapshot(self.ctxt, vol1.id)
self.assertRaises(exception.VolumeDriverException,
self.driver.create_snapshot,
snap)
self.driver.delete_volume(vol1)
# gmcv with specified cycle_period_seconds
vol1, model_update = self._create_test_volume(
self.gmcv_with_cps900_type)
self.assertEqual(fields.ReplicationStatus.ENABLED,
model_update['replication_status'])
self._validate_replic_vol_creation(vol1, True)
snap = testutils.create_snapshot(self.ctxt, vol1.id)
self.assertRaises(exception.VolumeDriverException,
self.driver.create_snapshot, snap)
self.driver.delete_volume(vol1)
def test_storwize_create_cloned_volume_with_mirror_replica(self): def test_storwize_create_cloned_volume_with_mirror_replica(self):
# Set replication target # Set replication target
self.driver.configuration.set_override('replication_device', self.driver.configuration.set_override('replication_device',

View File

@ -3582,16 +3582,6 @@ class StorwizeSVCCommonDriver(san.SanDriver,
LOG.error(msg) LOG.error(msg)
raise exception.VolumeDriverException(message=msg) raise exception.VolumeDriverException(message=msg)
rep_type = self._get_volume_replicated_type(
ctxt, None, source_vol['volume_type_id'])
if rep_type == storwize_const.GMCV:
# GMCV volume will have problem to failback
# when it has flash copy relationship besides change volumes
msg = _('create_snapshot: Create snapshot to '
'gmcv replication volume is not allowed.')
LOG.error(msg)
raise exception.VolumeDriverException(message=msg)
pool = volume_utils.extract_host(source_vol['host'], 'pool') pool = volume_utils.extract_host(source_vol['host'], 'pool')
opts = self._get_vdisk_params(source_vol['volume_type_id']) opts = self._get_vdisk_params(source_vol['volume_type_id'])
@ -6237,22 +6227,32 @@ class StorwizeSVCCommonDriver(san.SanDriver,
"relationship of %(vol)s does not exist in " "relationship of %(vol)s does not exist in "
"backend.", {'vol': volume.id}) "backend.", {'vol': volume.id})
model_update['status'] = fields.GroupStatus.ERROR model_update['status'] = fields.GroupStatus.ERROR
elif (rccg['copy_type'] != 'empty_group' and else:
(rccg['copy_type'] != rcrel['copy_type'] or if rccg and rccg.get('cycling_mode', None) == 'multi':
rccg['state'] != rcrel['state'] or self._helpers.stop_relationship(vol_name)
rccg['primary'] != rcrel['primary'] or rcrel = self._helpers.get_relationship_info(vol_name)
rccg['cycling_mode'] != rcrel['cycling_mode'] or if (rccg['state'] != 'empty' and
(rccg['cycle_period_seconds'] != rccg['state'] != 'consistent_stopped' or
rcrel['cycle_period_seconds']))): rccg['state'] != 'inconsistent_stopped'):
LOG.error("Failed to update rccg %(rccg)s: remote copy " self._helpers.stop_rccg(rccg_name)
"type of %(vol)s is %(vol_rc_type)s, the rccg " # To handle existing group updation, refresh rccg
"type is %(rccg_type)s. rcrel state is " # state to avoid unnecessary stop_rccg calls.
"%(rcrel_state)s, rccg state is %(rccg_state)s. " rccg = self._helpers.get_rccg(rccg_name)
"rcrel primary is %(rcrel_primary)s, rccg "
"primary is %(rccg_primary)s. " if (rccg['copy_type'] != 'empty_group' and
"rcrel cycling mode is %(rcrel_cmode)s, rccg " any(k for k in ('copy_type', 'state', 'primary',
"cycling mode is %(rccg_cmode)s. rcrel cycling " 'cycling_mode', 'cycle_period_seconds')
"period is %(rcrel_period)s, rccg cycling " if rccg[k] != rcrel[k])):
LOG.error("Failed to update rccg %(rccg)s: remote "
"copy type of %(vol)s is %(vol_rc_type)s, "
"the rccg type is %(rccg_type)s. rcrel "
"state %(rcrel_state)s, rccg state is "
"%(rccg_state)s rcrel primary is "
"%(rcrel_primary)s, rccg primary is "
"%(rccg_primary)s. rcrel cycling mode is "
"%(rcrel_cmode)s, rccg cycling mode is "
"%(rccg_cmode)s. rcrel cycling period is "
"%(rcrel_period)s, rccg cycling "
"period is %(rccg_period)s. ", "period is %(rccg_period)s. ",
{'rccg': rccg_name, {'rccg': rccg_name,
'vol': volume.id, 'vol': volume.id,
@ -6264,11 +6264,17 @@ class StorwizeSVCCommonDriver(san.SanDriver,
'rccg_primary': rccg['primary'], 'rccg_primary': rccg['primary'],
'rcrel_cmode': rcrel['cycling_mode'], 'rcrel_cmode': rcrel['cycling_mode'],
'rccg_cmode': rccg['cycling_mode'], 'rccg_cmode': rccg['cycling_mode'],
'rcrel_period': rcrel['cycle_period_seconds'], 'rcrel_period':
'rccg_period': rccg['cycle_period_seconds']}) rcrel['cycle_period_seconds'],
'rccg_period':
rccg['cycle_period_seconds']})
# This rcrel updation failed ,it has to be started
# explicitly.
self._helpers.start_relationship(vol_name)
model_update['status'] = fields.GroupStatus.ERROR model_update['status'] = fields.GroupStatus.ERROR
else: else:
self._helpers.chrcrelationship(rcrel['name'], rccg_name) self._helpers.chrcrelationship(rcrel['name'],
rccg_name)
if rccg['copy_type'] == 'empty_group': if rccg['copy_type'] == 'empty_group':
rccg = self._helpers.get_rccg(rccg_name) rccg = self._helpers.get_rccg(rccg_name)
added_vols.append({'id': volume.id, added_vols.append({'id': volume.id,
@ -6278,6 +6284,14 @@ class StorwizeSVCCommonDriver(san.SanDriver,
LOG.error("Failed to add the remote copy of volume %(vol)s to " LOG.error("Failed to add the remote copy of volume %(vol)s to "
"group. Exception: %(exception)s.", "group. Exception: %(exception)s.",
{'vol': volume.name, 'exception': err}) {'vol': volume.name, 'exception': err})
self._helpers.start_relationship(vol_name)
if (rccg and len(add_volumes) > 0 and
rccg.get('cycling_mode', None) == 'multi'):
if rccg.get('primary', None) == 'aux':
self._helpers.start_rccg(rccg_name, primary='aux')
elif rccg.get('primary', None) == 'master':
self._helpers.start_rccg(rccg_name, primary='master')
# Remove remote copy relationship from rccg # Remove remote copy relationship from rccg
removed_vols = [] removed_vols = []

View File

@ -0,0 +1,6 @@
---
fixes:
- |
IBM Spectrum Virtualize Family driver `Bug #1922013
<https://bugs.launchpad.net/cinder/+bug/1922013>`_: Fixed issues in
adding volumes to GMCV group.