XIV\A9000: Added replication group support
This patch adds group replication capabilities for the IBM driver for the A-line system. implements: * Enable replication on group * Disable replication on group * Fail over replication * Get replication error status Implements: blueprint replication-cg-ibm-storage Change-Id: Ic59c568502258096e24ca5ab81dc5b8cd1779995 Depends-On: Ia4af4dd011d569a3ac84387b37dcf2606da48fba
This commit is contained in:
		 Tzur Eliyahu
					Tzur Eliyahu
				
			
				
					committed by
					
						 Isaac Beckman
						Isaac Beckman
					
				
			
			
				
	
			
			
			 Isaac Beckman
						Isaac Beckman
					
				
			
						parent
						
							695c1f9272
						
					
				
				
					commit
					bb9a4e1a90
				
			| @@ -25,6 +25,7 @@ pyxcli_client.errors = fake_pyxcli_exceptions | |||||||
| pyxcli_client.events = mock.Mock() | pyxcli_client.events = mock.Mock() | ||||||
| pyxcli_client.mirroring = mock.Mock() | pyxcli_client.mirroring = mock.Mock() | ||||||
| pyxcli_client.transports = fake_pyxcli_exceptions | pyxcli_client.transports = fake_pyxcli_exceptions | ||||||
|  | pyxcli_client.mirroring.cg_recovery_manager = mock.Mock() | ||||||
|  |  | ||||||
| sys.modules['pyxcli'] = pyxcli_client | sys.modules['pyxcli'] = pyxcli_client | ||||||
| sys.modules['pyxcli.events'] = pyxcli_client.events | sys.modules['pyxcli.events'] = pyxcli_client.events | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ from xml.etree import ElementTree | |||||||
| from cinder import context | from cinder import context | ||||||
| from cinder import exception | from cinder import exception | ||||||
| from cinder import objects | from cinder import objects | ||||||
|  | from cinder.objects import fields | ||||||
| from cinder import test | from cinder import test | ||||||
| from cinder.tests.unit import fake_constants as fake | from cinder.tests.unit import fake_constants as fake | ||||||
| from cinder.tests.unit import utils as testutils | from cinder.tests.unit import utils as testutils | ||||||
| @@ -27,9 +28,11 @@ from cinder.tests.unit.volume.drivers.ibm import fake_pyxcli | |||||||
| import cinder.volume.drivers.ibm.ibm_storage as storage | import cinder.volume.drivers.ibm.ibm_storage as storage | ||||||
| from cinder.volume.drivers.ibm.ibm_storage import cryptish | from cinder.volume.drivers.ibm.ibm_storage import cryptish | ||||||
| from cinder.volume.drivers.ibm.ibm_storage.xiv_proxy import XIVProxy | from cinder.volume.drivers.ibm.ibm_storage.xiv_proxy import XIVProxy | ||||||
|  | from cinder.volume.drivers.ibm.ibm_storage import xiv_replication | ||||||
| from cinder.volume import group_types | from cinder.volume import group_types | ||||||
|  |  | ||||||
| errors = fake_pyxcli.pyxcli_client.errors | errors = fake_pyxcli.pyxcli_client.errors | ||||||
|  | mirroring = fake_pyxcli.pyxcli_client.mirroring | ||||||
|  |  | ||||||
| test_mock = mock.MagicMock() | test_mock = mock.MagicMock() | ||||||
| module_patcher = mock.MagicMock() | module_patcher = mock.MagicMock() | ||||||
| @@ -45,6 +48,11 @@ TEST_VOLUME = { | |||||||
|     'group_id': fake.CONSISTENCY_GROUP_ID, |     'group_id': fake.CONSISTENCY_GROUP_ID, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | TEST_GROUP_SPECS = { | ||||||
|  |     'group_replication_enabled': '<is> True', | ||||||
|  |     'replication_type': 'sync', | ||||||
|  | } | ||||||
|  |  | ||||||
| TEST_EXTRA_SPECS = { | TEST_EXTRA_SPECS = { | ||||||
|     'replication_enabled': '<is> False', |     'replication_enabled': '<is> False', | ||||||
| } | } | ||||||
| @@ -356,7 +364,174 @@ class XIVProxyTest(test.TestCase): | |||||||
|         ex = getattr(p, "_get_exception")() |         ex = getattr(p, "_get_exception")() | ||||||
|         self.assertRaises(ex, p.create_volume, volume) |         self.assertRaises(ex, p.create_volume, volume) | ||||||
|  |  | ||||||
|     @mock.patch("cinder.volume.utils.group_get_by_id", mock.MagicMock()) |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_replication.VolumeReplication.create_replication", | ||||||
|  |                 mock.MagicMock()) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_replication.GroupReplication.create_replication", | ||||||
|  |                 mock.MagicMock()) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy.get_group_specs_by_group_resource", | ||||||
|  |                 mock.MagicMock(return_value=(TEST_GROUP_SPECS, ''))) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy._get_target_params", | ||||||
|  |                 mock.MagicMock(return_value=REPLICA_PARAMS)) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy._get_target", | ||||||
|  |                 mock.MagicMock(return_value="BLABLA")) | ||||||
|  |     def test_enable_replication(self): | ||||||
|  |         """Test enable_replication""" | ||||||
|  |         driver = mock.MagicMock() | ||||||
|  |         driver.VERSION = "VERSION" | ||||||
|  |  | ||||||
|  |         p = self.proxy( | ||||||
|  |             self.default_storage_info, | ||||||
|  |             mock.MagicMock(), | ||||||
|  |             test_mock.cinder.exception, | ||||||
|  |             driver) | ||||||
|  |         p.ibm_storage_cli = mock.MagicMock() | ||||||
|  |         p._call_remote_xiv_xcli = mock.MagicMock() | ||||||
|  |         p._update_consistencygroup = mock.MagicMock() | ||||||
|  |         p.targets = {'tgt1': 'info1'} | ||||||
|  |  | ||||||
|  |         group = self._create_test_group('WTF') | ||||||
|  |         vol = testutils.create_volume(self.ctxt) | ||||||
|  |         ret = p.enable_replication(self.ctxt, group, [vol]) | ||||||
|  |  | ||||||
|  |         self.assertEqual(( | ||||||
|  |             {'replication_status': fields.ReplicationStatus.ENABLED}, | ||||||
|  |             [{'id': vol['id'], | ||||||
|  |               'replication_status': fields.ReplicationStatus.ENABLED}]), ret) | ||||||
|  |  | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_replication.VolumeReplication.delete_replication", | ||||||
|  |                 mock.MagicMock()) | ||||||
|  |     @mock.patch("cinder.volume.group_types.get_group_type_specs", | ||||||
|  |                 mock.MagicMock(return_value=TEST_GROUP_SPECS)) | ||||||
|  |     def test_disable_replication(self): | ||||||
|  |         """Test disable_replication""" | ||||||
|  |         driver = mock.MagicMock() | ||||||
|  |         driver.VERSION = "VERSION" | ||||||
|  |  | ||||||
|  |         p = self.proxy( | ||||||
|  |             self.default_storage_info, | ||||||
|  |             mock.MagicMock(), | ||||||
|  |             test_mock.cinder.exception, | ||||||
|  |             driver) | ||||||
|  |         p.ibm_storage_cli = mock.MagicMock() | ||||||
|  |         p._call_remote_xiv_xcli = mock.MagicMock() | ||||||
|  |         p._update_consistencygroup = mock.MagicMock() | ||||||
|  |  | ||||||
|  |         group = self._create_test_group('WTF') | ||||||
|  |         ret = p.disable_replication(self.ctxt, group, []) | ||||||
|  |  | ||||||
|  |         self.assertEqual(( | ||||||
|  |             {'replication_status': fields.ReplicationStatus.DISABLED}, []), | ||||||
|  |             ret) | ||||||
|  |  | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy._using_default_backend", | ||||||
|  |                 mock.MagicMock(return_value=False)) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy._get_target_params", | ||||||
|  |                 mock.MagicMock(return_value={'san_clustername': "master"})) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy._init_xcli", | ||||||
|  |                 mock.MagicMock()) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy._init_xcli", | ||||||
|  |                 mock.MagicMock()) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_proxy.XIVProxy.get_group_specs_by_group_resource", | ||||||
|  |                 mock.MagicMock(return_value=(TEST_GROUP_SPECS, ''))) | ||||||
|  |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|  |                 "xiv_replication.GroupReplication.failover", | ||||||
|  |                 mock.MagicMock(return_value=(True, 'good'))) | ||||||
|  |     def test_failover_replication_with_default(self): | ||||||
|  |         driver = mock.MagicMock() | ||||||
|  |         driver.VERSION = "VERSION" | ||||||
|  |  | ||||||
|  |         p = self.proxy( | ||||||
|  |             self.default_storage_info, | ||||||
|  |             mock.MagicMock(), | ||||||
|  |             test_mock.cinder.exception, | ||||||
|  |             driver) | ||||||
|  |         group = self._create_test_group('WTF') | ||||||
|  |         group.replication_status = fields.ReplicationStatus.FAILED_OVER | ||||||
|  |         vol = testutils.create_volume(self.ctxt) | ||||||
|  |         group_update, vol_update = p.failover_replication(self.ctxt, group, | ||||||
|  |                                                           [vol], 'default') | ||||||
|  |         updates = {'status': 'available'} | ||||||
|  |         self.assertEqual(({'replication_status': 'available'}, | ||||||
|  |                           [{'volume_id': vol['id'], | ||||||
|  |                             'updates': updates}]), (group_update, vol_update)) | ||||||
|  |  | ||||||
|  |     def test_failover_resource_no_mirror(self): | ||||||
|  |         driver = mock.MagicMock() | ||||||
|  |         driver.VERSION = "VERSION" | ||||||
|  |  | ||||||
|  |         p = self.proxy( | ||||||
|  |             self.default_storage_info, | ||||||
|  |             mock.MagicMock(), | ||||||
|  |             test_mock.cinder.exception, | ||||||
|  |             driver) | ||||||
|  |  | ||||||
|  |         recovery_mgr = mock.MagicMock() | ||||||
|  |         recovery_mgr.is_mirror_active = mock.MagicMock() | ||||||
|  |         recovery_mgr.is_mirror_active.return_value = False | ||||||
|  |  | ||||||
|  |         group = self._create_test_group('WTF') | ||||||
|  |         ret = xiv_replication.Replication(p)._failover_resource( | ||||||
|  |             group, recovery_mgr, mock.MagicMock, 'cg', True) | ||||||
|  |         msg = ("%(rep_type)s %(res)s: no active mirroring and can not " | ||||||
|  |                "failback" % {'rep_type': 'cg', | ||||||
|  |                              'res': group['name']}) | ||||||
|  |         self.assertEqual((False, msg), ret) | ||||||
|  |  | ||||||
|  |     def test_failover_resource_mirror(self): | ||||||
|  |         driver = mock.MagicMock() | ||||||
|  |         driver.VERSION = "VERSION" | ||||||
|  |  | ||||||
|  |         p = self.proxy( | ||||||
|  |             self.default_storage_info, | ||||||
|  |             mock.MagicMock(), | ||||||
|  |             test_mock.cinder.exception, | ||||||
|  |             driver) | ||||||
|  |         recovery_mgr = mock.MagicMock() | ||||||
|  |         recovery_mgr.is_mirror_active = mock.MagicMock() | ||||||
|  |         recovery_mgr.is_mirror_active.return_value = True | ||||||
|  |  | ||||||
|  |         group = self._create_test_group('WTF') | ||||||
|  |         ret = xiv_replication.Replication(p)._failover_resource( | ||||||
|  |             group, recovery_mgr, mock.MagicMock, 'cg', True) | ||||||
|  |  | ||||||
|  |         self.assertEqual((True, None), ret) | ||||||
|  |  | ||||||
|  |     def test_failover_resource_change_role(self): | ||||||
|  |         driver = mock.MagicMock() | ||||||
|  |         driver.VERSION = "VERSION" | ||||||
|  |  | ||||||
|  |         p = self.proxy( | ||||||
|  |             self.default_storage_info, | ||||||
|  |             mock.MagicMock(), | ||||||
|  |             test_mock.cinder.exception, | ||||||
|  |             driver) | ||||||
|  |         recovery_mgr = mock.MagicMock() | ||||||
|  |         recovery_mgr.is_mirror_active = mock.MagicMock() | ||||||
|  |         recovery_mgr.is_mirror_active.return_value = True | ||||||
|  |         recovery_mgr.switch_roles.side_effect = ( | ||||||
|  |             errors.XCLIError('')) | ||||||
|  |         failover_rep_mgr = mock.MagicMock() | ||||||
|  |         failover_rep_mgr.change_role = mock.MagicMock() | ||||||
|  |         group = self._create_test_group('WTF') | ||||||
|  |  | ||||||
|  |         xiv_replication.Replication(p)._failover_resource( | ||||||
|  |             group, recovery_mgr, failover_rep_mgr, 'cg', True) | ||||||
|  |  | ||||||
|  |         failover_rep_mgr.change_role.assert_called_once_with( | ||||||
|  |             resource_id=group['name'], | ||||||
|  |             new_role='Slave') | ||||||
|  |  | ||||||
|     @mock.patch("cinder.volume.utils.is_group_a_cg_snapshot_type", |     @mock.patch("cinder.volume.utils.is_group_a_cg_snapshot_type", | ||||||
|                 mock.MagicMock(return_value=True)) |                 mock.MagicMock(return_value=True)) | ||||||
|     def test_create_volume_with_consistency_group(self): |     def test_create_volume_with_consistency_group(self): | ||||||
| @@ -376,8 +551,8 @@ class XIVProxyTest(test.TestCase): | |||||||
|         vol_type = testutils.create_volume_type(self.ctxt, name='WTF') |         vol_type = testutils.create_volume_type(self.ctxt, name='WTF') | ||||||
|         volume = testutils.create_volume( |         volume = testutils.create_volume( | ||||||
|             self.ctxt, size=16, volume_type_id=vol_type.id) |             self.ctxt, size=16, volume_type_id=vol_type.id) | ||||||
|         grp = testutils.create_group(self.ctxt, name='bla', group_type_id='1', |  | ||||||
|                                      volume_type_ids=[vol_type.id]) |         grp = self._create_test_group('WTF') | ||||||
|         volume.group = grp |         volume.group = grp | ||||||
|         p.create_volume(volume) |         p.create_volume(volume) | ||||||
|  |  | ||||||
| @@ -390,7 +565,7 @@ class XIVProxyTest(test.TestCase): | |||||||
|             cg='cg') |             cg='cg') | ||||||
|  |  | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|                 "xiv_proxy.XIVProxy._replication_create", |                 "xiv_replication.VolumeReplication.create_replication", | ||||||
|                 mock.MagicMock()) |                 mock.MagicMock()) | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|                 "xiv_proxy.XIVProxy._get_qos_specs", |                 "xiv_proxy.XIVProxy._get_qos_specs", | ||||||
| @@ -417,7 +592,7 @@ class XIVProxyTest(test.TestCase): | |||||||
|         p.create_volume(volume) |         p.create_volume(volume) | ||||||
|  |  | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|                 "xiv_proxy.XIVProxy._replication_create", |                 "xiv_replication.VolumeReplication.create_replication", | ||||||
|                 mock.MagicMock()) |                 mock.MagicMock()) | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|                 "xiv_proxy.XIVProxy._get_qos_specs", |                 "xiv_proxy.XIVProxy._get_qos_specs", | ||||||
| @@ -446,10 +621,6 @@ class XIVProxyTest(test.TestCase): | |||||||
|         ex = getattr(p, "_get_exception")() |         ex = getattr(p, "_get_exception")() | ||||||
|         self.assertRaises(ex, p.create_volume, volume) |         self.assertRaises(ex, p.create_volume, volume) | ||||||
|  |  | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |  | ||||||
|                 "xiv_proxy.XIVProxy._get_targets", |  | ||||||
|                 mock.MagicMock( |  | ||||||
|                     return_value={'tgt1': 'info1', 'tgt2': 'info2'})) |  | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|                 "xiv_proxy.XIVProxy._get_qos_specs", |                 "xiv_proxy.XIVProxy._get_qos_specs", | ||||||
|                 mock.MagicMock(return_value=None)) |                 mock.MagicMock(return_value=None)) | ||||||
| @@ -495,7 +666,7 @@ class XIVProxyTest(test.TestCase): | |||||||
|         p.ibm_storage_cli.cmd.vol_delete.assert_called_once_with(vol='WTF32') |         p.ibm_storage_cli.cmd.vol_delete.assert_called_once_with(vol='WTF32') | ||||||
|  |  | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|                 "xiv_proxy.XIVProxy._replication_delete", |                 "xiv_replication.VolumeReplication.delete_replication", | ||||||
|                 mock.MagicMock()) |                 mock.MagicMock()) | ||||||
|     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage." | ||||||
|                 "xiv_proxy.XIVProxy._get_extra_specs", |                 "xiv_proxy.XIVProxy._get_extra_specs", | ||||||
| @@ -1406,7 +1577,9 @@ class XIVProxyTest(test.TestCase): | |||||||
|         ex = getattr(p, "_get_exception")() |         ex = getattr(p, "_get_exception")() | ||||||
|         self.assertRaises(ex, p.create_group, {}, self._create_test_group()) |         self.assertRaises(ex, p.create_group, {}, self._create_test_group()) | ||||||
|  |  | ||||||
|     def test_create_consistencygroup_with_replication(self): |     @mock.patch("cinder.volume.drivers.ibm.ibm_storage.xiv_proxy." | ||||||
|  |                 "client.XCLIClient") | ||||||
|  |     def test_create_consistencygroup_with_replication(self, mock_xcli): | ||||||
|         """test create_consistenygroup when replication is set""" |         """test create_consistenygroup when replication is set""" | ||||||
|  |  | ||||||
|         p = self.proxy( |         p = self.proxy( | ||||||
| @@ -1426,8 +1599,8 @@ class XIVProxyTest(test.TestCase): | |||||||
|         group_obj.volume_types = objects.VolumeTypeList(context=self.ctxt, |         group_obj.volume_types = objects.VolumeTypeList(context=self.ctxt, | ||||||
|                                                         objects=[vol_type]) |                                                         objects=[vol_type]) | ||||||
|  |  | ||||||
|         ex = getattr(p, "_get_exception")() |         model_update = p.create_group({}, group_obj) | ||||||
|         self.assertRaises(ex, p.create_group, {}, group_obj) |         self.assertEqual('available', model_update['status']) | ||||||
|  |  | ||||||
|     def test_create_consistencygroup_from_src_cgsnapshot(self): |     def test_create_consistencygroup_from_src_cgsnapshot(self): | ||||||
|         """test a successful cg create from cgsnapshot""" |         """test a successful cg create from cgsnapshot""" | ||||||
| @@ -2113,7 +2286,7 @@ class XIVProxyTest(test.TestCase): | |||||||
|             test_mock.cinder.exception, |             test_mock.cinder.exception, | ||||||
|             driver) |             driver) | ||||||
|  |  | ||||||
|         p._replication_create = test_mock.MagicMock(return_value=None) |         xiv_replication.VolumeReplication = mock.MagicMock() | ||||||
|         grp = testutils.create_group(self.ctxt, name='bla', group_type_id='1') |         grp = testutils.create_group(self.ctxt, name='bla', group_type_id='1') | ||||||
|         volume = testutils.create_volume(self.ctxt, display_name='bla') |         volume = testutils.create_volume(self.ctxt, display_name='bla') | ||||||
|         volume.group = grp |         volume.group = grp | ||||||
|   | |||||||
| @@ -365,14 +365,6 @@ class IBMStorageProxy(object): | |||||||
|         except Exception: |         except Exception: | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|     def _get_targets(self): |  | ||||||
|         return self.targets |  | ||||||
|  |  | ||||||
|     def _is_replication_supported(self): |  | ||||||
|         if self.targets: |  | ||||||
|             return True |  | ||||||
|         return False |  | ||||||
|  |  | ||||||
|     @_trace_time |     @_trace_time | ||||||
|     def _read_replication_devices(self): |     def _read_replication_devices(self): | ||||||
|         """Read replication devices from configuration |         """Read replication devices from configuration | ||||||
|   | |||||||
| @@ -27,8 +27,7 @@ if pyxcli: | |||||||
|     from pyxcli import client |     from pyxcli import client | ||||||
|     from pyxcli import errors |     from pyxcli import errors | ||||||
|     from pyxcli.events import events |     from pyxcli.events import events | ||||||
|     from pyxcli.mirroring import errors as m_errors |     from pyxcli.mirroring import mirrored_entities | ||||||
|     from pyxcli.mirroring import volume_recovery_manager |  | ||||||
|     from pyxcli import transports |     from pyxcli import transports | ||||||
|  |  | ||||||
| from cinder import context | from cinder import context | ||||||
| @@ -40,6 +39,8 @@ from cinder.volume.drivers.ibm.ibm_storage import certificate | |||||||
| from cinder.volume.drivers.ibm.ibm_storage import cryptish | from cinder.volume.drivers.ibm.ibm_storage import cryptish | ||||||
| from cinder.volume.drivers.ibm.ibm_storage import proxy | from cinder.volume.drivers.ibm.ibm_storage import proxy | ||||||
| from cinder.volume.drivers.ibm.ibm_storage import strings | from cinder.volume.drivers.ibm.ibm_storage import strings | ||||||
|  | from cinder.volume.drivers.ibm.ibm_storage import xiv_replication as repl | ||||||
|  | from cinder.volume import group_types | ||||||
| from cinder.volume import qos_specs | from cinder.volume import qos_specs | ||||||
| from cinder.volume import utils | from cinder.volume import utils | ||||||
| from cinder.volume import volume_types | from cinder.volume import volume_types | ||||||
| @@ -95,38 +96,19 @@ MANAGE_VOLUME_BASE_ERROR = _("Unable to manage the volume '%(volume)s': " | |||||||
|                              "%(error)s.") |                              "%(error)s.") | ||||||
|  |  | ||||||
|  |  | ||||||
| class Rate(object): |  | ||||||
|  |  | ||||||
|     def __init__(self, rpo, schedule): |  | ||||||
|         self.rpo = rpo |  | ||||||
|         self.schedule = schedule |  | ||||||
|         self.schedule_name = self._schedule_name_from_schedule(self.schedule) |  | ||||||
|  |  | ||||||
|     def _schedule_name_from_schedule(self, schedule): |  | ||||||
|         if schedule == '00:00:20': |  | ||||||
|             return 'min_interval' |  | ||||||
|         return ("cinder_%(sched)s" % |  | ||||||
|                 {'sched': schedule.replace(':', '_')}) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class XIVProxy(proxy.IBMStorageProxy): | class XIVProxy(proxy.IBMStorageProxy): | ||||||
|     """Proxy between the Cinder Volume and Spectrum Accelerate Storage. |     """Proxy between the Cinder Volume and Spectrum Accelerate Storage. | ||||||
|  |  | ||||||
|     Supports IBM XIV, Spectrum Accelerate, A9000, A9000R |     Supports IBM XIV, Spectrum Accelerate, A9000, A9000R | ||||||
|     Version: 2.1.0 |     Version: 2.1.0 | ||||||
|     Required pyxcli version: 1.1.2 |     Required pyxcli version: 1.1.4 | ||||||
|  |  | ||||||
|     2.0 - First open source driver version |     2.0 - First open source driver version | ||||||
|     2.1.0 - Support Consistency groups through Generic volume groups |     2.1.0 - Support Consistency groups through Generic volume groups | ||||||
|           - Support XIV/A9000 Volume independent QoS |           - Support XIV/A9000 Volume independent QoS | ||||||
|  |           - Support groups replication | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     async_rates = ( |  | ||||||
|         Rate(rpo=120, schedule='00:01:00'), |  | ||||||
|         Rate(rpo=300, schedule='00:02:00'), |  | ||||||
|         Rate(rpo=600, schedule='00:05:00'), |  | ||||||
|         Rate(rpo=1200, schedule='00:10:00'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def __init__(self, storage_info, logger, exception, |     def __init__(self, storage_info, logger, exception, | ||||||
|                  driver=None, active_backend_id=None): |                  driver=None, active_backend_id=None): | ||||||
| @@ -192,13 +174,6 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         LOG.info("Connection to the IBM storage " |         LOG.info("Connection to the IBM storage " | ||||||
|                  "system established successfully.") |                  "system established successfully.") | ||||||
|  |  | ||||||
|     def _get_schedule_from_rpo(self, rpo): |  | ||||||
|         return [rate for rate in self.async_rates |  | ||||||
|                 if rate.rpo == rpo][0].schedule_name |  | ||||||
|  |  | ||||||
|     def _get_supported_rpo(self): |  | ||||||
|         return [rate.rpo for rate in self.async_rates] |  | ||||||
|  |  | ||||||
|     @proxy._trace_time |     @proxy._trace_time | ||||||
|     def _update_active_schedule_objects(self): |     def _update_active_schedule_objects(self): | ||||||
|         """Set schedule objects on active backend. |         """Set schedule objects on active backend. | ||||||
| @@ -207,7 +182,7 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         min_interval. |         min_interval. | ||||||
|         """ |         """ | ||||||
|         schedules = self._call_xiv_xcli("schedule_list").as_dict('name') |         schedules = self._call_xiv_xcli("schedule_list").as_dict('name') | ||||||
|         for rate in self.async_rates: |         for rate in repl.Replication.async_rates: | ||||||
|             if rate.schedule == '00:00:20': |             if rate.schedule == '00:00:20': | ||||||
|                 continue |                 continue | ||||||
|             name = rate.schedule_name |             name = rate.schedule_name | ||||||
| @@ -245,7 +220,7 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         min_interval. |         min_interval. | ||||||
|         """ |         """ | ||||||
|         schedules = self._call_remote_xiv_xcli("schedule_list").as_dict('name') |         schedules = self._call_remote_xiv_xcli("schedule_list").as_dict('name') | ||||||
|         for rate in self.async_rates: |         for rate in repl.Replication.async_rates: | ||||||
|             if rate.schedule == '00:00:20': |             if rate.schedule == '00:00:20': | ||||||
|                 continue |                 continue | ||||||
|             name = rate.schedule_name |             name = rate.schedule_name | ||||||
| @@ -438,30 +413,12 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         return self._get_qos_specs(type_id) |         return self._get_qos_specs(type_id) | ||||||
|  |  | ||||||
|     def _get_replication_info(self, specs): |     def _get_replication_info(self, specs): | ||||||
|         info = {'enabled': False, 'mode': None, 'rpo': 0} |  | ||||||
|         if specs: |         info, msg = repl.Replication.extract_replication_info_from_specs(specs) | ||||||
|             LOG.debug('_get_replication_info: specs %(specs)s', |         if not info: | ||||||
|                       {'specs': specs}) |  | ||||||
|             info['enabled'] = ( |  | ||||||
|                 specs.get('replication_enabled', '').upper() in |  | ||||||
|                 (u'TRUE', strings.METADATA_IS_TRUE)) |  | ||||||
|             replication_type = specs.get('replication_type', SYNC).lower() |  | ||||||
|             if replication_type in (u'sync', u'<is> sync'): |  | ||||||
|                 info['mode'] = SYNC |  | ||||||
|             elif replication_type in (u'async', u'<is> async'): |  | ||||||
|                 info['mode'] = ASYNC |  | ||||||
|             else: |  | ||||||
|                 msg = (_("Unsupported replication mode %(mode)s") |  | ||||||
|                        % {'mode': replication_type}) |  | ||||||
|             LOG.error(msg) |             LOG.error(msg) | ||||||
|             raise self._get_exception()(message=msg) |             raise self._get_exception()(message=msg) | ||||||
|             info['rpo'] = int(specs.get('rpo', u'<is> 0')[5:]) |  | ||||||
|             if info['rpo'] and info['rpo'] not in self._get_supported_rpo(): |  | ||||||
|                 msg = (_("Unsupported replication RPO %(rpo)s") |  | ||||||
|                        % {'rpo': info['rpo']}) |  | ||||||
|                 LOG.error(msg) |  | ||||||
|                 raise self._get_exception()(message=msg) |  | ||||||
|             LOG.debug('_get_replication_info: info %(info)s', {'info': info}) |  | ||||||
|         return info |         return info | ||||||
|  |  | ||||||
|     @proxy._trace_time |     @proxy._trace_time | ||||||
| @@ -491,26 +448,74 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|     @proxy._trace_time |     @proxy._trace_time | ||||||
|     def create_volume(self, volume): |     def create_volume(self, volume): | ||||||
|         """Creates a volume.""" |         """Creates a volume.""" | ||||||
|  |  | ||||||
|         # read replication information |         # read replication information | ||||||
|         specs = self._get_extra_specs(volume.get('volume_type_id', None)) |         specs = self._get_extra_specs(volume.get('volume_type_id', None)) | ||||||
|         replication_info = self._get_replication_info(specs) |         replication_info = self._get_replication_info(specs) | ||||||
|  |  | ||||||
|         if volume.group and replication_info['enabled']: |  | ||||||
|             # An unsupported illegal configuration |  | ||||||
|             msg = _("Unable to create volume: " |  | ||||||
|                     "Replication of consistency group is not supported") |  | ||||||
|             LOG.error(msg) |  | ||||||
|             raise self.meta['exception'].VolumeBackendAPIException(data=msg) |  | ||||||
|  |  | ||||||
|         self._create_volume(volume) |         self._create_volume(volume) | ||||||
|         return self.handle_created_vol_properties(replication_info, |         return self.handle_created_vol_properties(replication_info, | ||||||
|                                                   volume) |                                                   volume) | ||||||
|  |  | ||||||
|     def handle_created_vol_properties(self, replication_info, volume): |     def handle_created_vol_properties(self, replication_info, volume): | ||||||
|         volume_update = {} |         volume_update = {} | ||||||
|  |  | ||||||
|  |         LOG.debug('checking replication_info %(rep)s', | ||||||
|  |                   {'rep': replication_info}) | ||||||
|  |         volume_update['replication_status'] = 'disabled' | ||||||
|         cg = volume.group and utils.is_group_a_cg_snapshot_type(volume.group) |         cg = volume.group and utils.is_group_a_cg_snapshot_type(volume.group) | ||||||
|  |         if replication_info['enabled']: | ||||||
|  |             try: | ||||||
|  |                 repl.VolumeReplication(self).create_replication( | ||||||
|  |                     volume.name, replication_info) | ||||||
|  |             except Exception as e: | ||||||
|  |                 details = self._get_code_and_status_or_message(e) | ||||||
|  |                 msg = ('Failed create_replication for ' | ||||||
|  |                        'volume %(vol)s: %(err)s', | ||||||
|  |                        {'vol': volume['name'], 'err': details}) | ||||||
|  |                 LOG.error(msg) | ||||||
|                 if cg: |                 if cg: | ||||||
|  |                     cg_name = self._cg_name_from_volume(volume) | ||||||
|  |                     self._silent_delete_volume_from_cg(volume, cg_name) | ||||||
|  |                 self._silent_delete_volume(volume=volume) | ||||||
|  |                 raise | ||||||
|  |             volume_update['replication_status'] = 'enabled' | ||||||
|  |  | ||||||
|  |         if cg: | ||||||
|  |             if volume.group.is_replicated: | ||||||
|  |                 # for replicated Consistency Group: | ||||||
|  |                 # The Volume must be mirrored, and its mirroring settings must | ||||||
|  |                 # be identical to those of the Consistency Group: | ||||||
|  |                 # mirroring type (e.g., synchronous), | ||||||
|  |                 # mirroring status, mirroring target(backend) | ||||||
|  |                 group_specs = group_types.get_group_type_specs( | ||||||
|  |                     volume.group.group_type_id) | ||||||
|  |                 group_rep_info = self._get_replication_info(group_specs) | ||||||
|  |  | ||||||
|  |                 msg = None | ||||||
|  |                 if volume_update['replication_status'] != 'enabled': | ||||||
|  |                     msg = ('Cannot add non-replicated volume into' | ||||||
|  |                            ' replicated group') | ||||||
|  |                 elif replication_info['mode'] != group_rep_info['mode']: | ||||||
|  |                     msg = ('Volume replication type and Group replication type' | ||||||
|  |                            ' should be the same') | ||||||
|  |                 elif volume.host != volume.group.host: | ||||||
|  |                     msg = 'Cannot add volume to Group on different host' | ||||||
|  |                 else: | ||||||
|  |                     group_name = self._cg_name_from_group(volume.group) | ||||||
|  |                     me = mirrored_entities.MirroredEntities( | ||||||
|  |                         self.ibm_storage_cli) | ||||||
|  |                     me_objs = me.get_mirror_resources_by_name_map() | ||||||
|  |                     vol_sync_state = me_objs['volumes'][volume.name].sync_state | ||||||
|  |                     cg_sync_state = me_objs['cgs'][group_name].sync_state | ||||||
|  |  | ||||||
|  |                     if (vol_sync_state != 'Synchronized' or | ||||||
|  |                             cg_sync_state != 'Synchronized'): | ||||||
|  |                         msg = ('Cannot add volume to Group. Both volume and ' | ||||||
|  |                                'group should have sync_state = Synchronized') | ||||||
|  |                 if msg: | ||||||
|  |                     LOG.error(msg) | ||||||
|  |                     raise self.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                         data=msg) | ||||||
|             try: |             try: | ||||||
|                 cg_name = self._cg_name_from_volume(volume) |                 cg_name = self._cg_name_from_volume(volume) | ||||||
|                 self._call_xiv_xcli( |                 self._call_xiv_xcli( | ||||||
| @@ -543,35 +548,234 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|                 raise self.meta['exception'].VolumeBackendAPIException( |                 raise self.meta['exception'].VolumeBackendAPIException( | ||||||
|                     data=msg) |                     data=msg) | ||||||
|  |  | ||||||
|         LOG.debug('checking replication_info %(rep)s', |  | ||||||
|                   {'rep': replication_info}) |  | ||||||
|         volume_update['replication_status'] = 'disabled' |  | ||||||
|         if replication_info['enabled']: |  | ||||||
|             try: |  | ||||||
|                 self._replication_create(volume, replication_info) |  | ||||||
|             except Exception as e: |  | ||||||
|                 details = self._get_code_and_status_or_message(e) |  | ||||||
|                 msg = ('Failed _replication_create for ' |  | ||||||
|                        'volume %(vol)s: %(err)s', |  | ||||||
|                        {'vol': volume['name'], 'err': details}) |  | ||||||
|                 LOG.error(msg) |  | ||||||
|                 if cg: |  | ||||||
|                     cg_name = self._cg_name_from_volume(volume) |  | ||||||
|                     self._silent_delete_volume_from_cg(volume, cg_name) |  | ||||||
|                 self._silent_delete_volume(volume=volume) |  | ||||||
|                 raise |  | ||||||
|             volume_update['replication_status'] = 'enabled' |  | ||||||
|  |  | ||||||
|         return volume_update |         return volume_update | ||||||
|  |  | ||||||
|  |     def get_group_specs_by_group_resource(self, context, group): | ||||||
|  |         group_type = group.get('group_type_id', None) | ||||||
|  |         if group_type is None: | ||||||
|  |             msg = ('No group specs inside group type.') | ||||||
|  |             return None, msg | ||||||
|  |         group_specs = group_types.get_group_type_specs(group_type) | ||||||
|  |         keyword = 'consistent_group_replication_enabled' | ||||||
|  |         if not group_specs.get(keyword) == '<is> True': | ||||||
|  |             msg = ('No cg replication field in group specs.') | ||||||
|  |             return None, msg | ||||||
|  |         return group_specs, '' | ||||||
|  |  | ||||||
|  |     @proxy._trace_time | ||||||
|  |     def enable_replication(self, context, group, volumes): | ||||||
|  |         """Enable cg replication""" | ||||||
|  |         # fetch replication info | ||||||
|  |         group_specs = group_types.get_group_type_specs(group.group_type_id) | ||||||
|  |         if not group_specs: | ||||||
|  |             msg = 'No group specs inside group type' | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.meta['exception'].VolumeBackendAPIException(data=msg) | ||||||
|  |  | ||||||
|  |         # Add this field to adjust it to generic replication (for volumes) | ||||||
|  |         replication_info = self._get_replication_info(group_specs) | ||||||
|  |         if utils.is_group_a_cg_snapshot_type(group): | ||||||
|  |             # take every vol out of cg - we can't mirror the cg otherwise. | ||||||
|  |             if volumes: | ||||||
|  |                 self._update_consistencygroup(context, group, | ||||||
|  |                                               remove_volumes=volumes) | ||||||
|  |                 for volume in volumes: | ||||||
|  |                     repl.VolumeReplication(self).create_replication( | ||||||
|  |                         volume.name, replication_info) | ||||||
|  |  | ||||||
|  |             # mirror entire group | ||||||
|  |             group_name = self._cg_name_from_group(group) | ||||||
|  |             self._create_consistencygroup_on_remote(context, group_name) | ||||||
|  |             repl.GroupReplication(self).create_replication(group_name, | ||||||
|  |                                                            replication_info) | ||||||
|  |  | ||||||
|  |             updated_volumes = [] | ||||||
|  |             if volumes: | ||||||
|  |                 # add volumes back to cg | ||||||
|  |                 self._update_consistencygroup(context, group, | ||||||
|  |                                               add_volumes=volumes) | ||||||
|  |                 for volume in volumes: | ||||||
|  |                     updated_volumes.append( | ||||||
|  |                         {'id': volume['id'], | ||||||
|  |                          'replication_status': | ||||||
|  |                              fields.ReplicationStatus.ENABLED}) | ||||||
|  |             return ({'replication_status': fields.ReplicationStatus.ENABLED}, | ||||||
|  |                     updated_volumes) | ||||||
|  |         else: | ||||||
|  |             # For generic groups we replicate all the volumes | ||||||
|  |             updated_volumes = [] | ||||||
|  |             for volume in volumes: | ||||||
|  |                 repl.VolumeReplication(self).create_replication( | ||||||
|  |                     volume.name, replication_info) | ||||||
|  |  | ||||||
|  |             # update status | ||||||
|  |             for volume in volumes: | ||||||
|  |                 updated_volumes.append( | ||||||
|  |                     {'id': volume['id'], | ||||||
|  |                      'replication_status': fields.ReplicationStatus.ENABLED}) | ||||||
|  |             return ({'replication_status': fields.ReplicationStatus.ENABLED}, | ||||||
|  |                     updated_volumes) | ||||||
|  |  | ||||||
|  |     @proxy._trace_time | ||||||
|  |     def disable_replication(self, context, group, volumes): | ||||||
|  |         """disables CG replication""" | ||||||
|  |         group_specs = group_types.get_group_type_specs(group.group_type_id) | ||||||
|  |         if not group_specs: | ||||||
|  |             msg = 'No group specs inside group type' | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.meta['exception'].VolumeBackendAPIException(data=msg) | ||||||
|  |  | ||||||
|  |         replication_info = self._get_replication_info(group_specs) | ||||||
|  |         updated_volumes = [] | ||||||
|  |         if utils.is_group_a_cg_snapshot_type(group): | ||||||
|  |             # one call deletes replication for cgs and volumes together. | ||||||
|  |             repl.GroupReplication(self).delete_replication(group, | ||||||
|  |                                                            replication_info) | ||||||
|  |             for volume in volumes: | ||||||
|  |                 # xiv locks volumes after deletion of replication. | ||||||
|  |                 # we need to unlock it for further use. | ||||||
|  |                 try: | ||||||
|  |                     self.ibm_storage_cli.cmd.vol_unlock(vol=volume.name) | ||||||
|  |                 except errors.XCLIError as e: | ||||||
|  |                     details = self._get_code_and_status_or_message(e) | ||||||
|  |                     msg = ('Failed to unlock volumes %(details)s' % | ||||||
|  |                            {'details': details}) | ||||||
|  |                     LOG.error(msg) | ||||||
|  |                     raise self.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                         data=msg) | ||||||
|  |                 updated_volumes.append( | ||||||
|  |                     {'id': volume.id, | ||||||
|  |                      'replication_status': fields.ReplicationStatus.DISABLED}) | ||||||
|  |         else: | ||||||
|  |             # For generic groups we replicate all the volumes | ||||||
|  |             updated_volumes = [] | ||||||
|  |             for volume in volumes: | ||||||
|  |                 repl.VolumeReplication(self).delete_replication( | ||||||
|  |                     volume.name, replication_info) | ||||||
|  |  | ||||||
|  |             # update status | ||||||
|  |             for volume in volumes: | ||||||
|  |                 updated_volumes.append( | ||||||
|  |                     {'id': volume['id'], | ||||||
|  |                      'replication_status': fields.ReplicationStatus.DISABLED}) | ||||||
|  |         return ({'replication_status': fields.ReplicationStatus.DISABLED}, | ||||||
|  |                 updated_volumes) | ||||||
|  |  | ||||||
|  |     def get_secondary_backend_id(self, secondary_backend_id): | ||||||
|  |         if secondary_backend_id is None: | ||||||
|  |             secondary_backend_id = self._get_target() | ||||||
|  |         if secondary_backend_id is None: | ||||||
|  |             msg = _("No targets defined. Can't perform failover.") | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                 data=msg) | ||||||
|  |         return secondary_backend_id | ||||||
|  |  | ||||||
|  |     def check_for_splitbrain(self, volumes, pool_master, pool_slave): | ||||||
|  |         if volumes: | ||||||
|  |             # check for split brain situations | ||||||
|  |             # check for files that are available on both volumes | ||||||
|  |             # and are not in an active mirroring relation | ||||||
|  |             split_brain = self._potential_split_brain( | ||||||
|  |                 self.ibm_storage_cli, | ||||||
|  |                 self.ibm_storage_remote_cli, | ||||||
|  |                 volumes, pool_master, | ||||||
|  |                 pool_slave) | ||||||
|  |             if split_brain: | ||||||
|  |                 # if such a situation exists stop and raise an exception! | ||||||
|  |                 msg = (_("A potential split brain condition has been found " | ||||||
|  |                          "with the following volumes: \n'%(volumes)s.'") % | ||||||
|  |                        {'volumes': split_brain}) | ||||||
|  |                 LOG.error(msg) | ||||||
|  |                 raise self.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                     data=msg) | ||||||
|  |  | ||||||
|  |     def failover_replication(self, context, group, volumes, | ||||||
|  |                              secondary_backend_id): | ||||||
|  |         """Failover a cg with all it's volumes. | ||||||
|  |  | ||||||
|  |         if secondery_id is default, cg needs to be failed back. | ||||||
|  |  | ||||||
|  |         """ | ||||||
|  |         volumes_updated = [] | ||||||
|  |         goal_status = '' | ||||||
|  |         pool_master = None | ||||||
|  |         group_updated = {'replication_status': group.replication_status} | ||||||
|  |         LOG.info("failover_replication: of cg %(cg)s " | ||||||
|  |                  "from %(active)s to %(id)s", | ||||||
|  |                  {'cg': group.get('name'), | ||||||
|  |                   'active': self.active_backend_id, | ||||||
|  |                   'id': secondary_backend_id}) | ||||||
|  |         if secondary_backend_id == strings.PRIMARY_BACKEND_ID: | ||||||
|  |             # default as active backend id | ||||||
|  |             if self._using_default_backend(): | ||||||
|  |                 LOG.info("CG has been failed back. " | ||||||
|  |                          "No need to fail back again.") | ||||||
|  |                 return group_updated, volumes_updated | ||||||
|  |             # get the master pool, not using default id. | ||||||
|  |             pool_master = self._get_target_params( | ||||||
|  |                 self.active_backend_id)['san_clustername'] | ||||||
|  |             pool_slave = self.storage_info[storage.FLAG_KEYS['storage_pool']] | ||||||
|  |             goal_status = 'available' | ||||||
|  |         else: | ||||||
|  |             if self._using_default_backend(): | ||||||
|  |                 LOG.info("cg already failed over.") | ||||||
|  |                 return group_updated, volumes_updated | ||||||
|  |             # using same api as Cheesecake, we need | ||||||
|  |             # replciation_device entry. so we use get_targets. | ||||||
|  |             secondary_backend_id = self.get_secondary_backend_id( | ||||||
|  |                 secondary_backend_id) | ||||||
|  |             pool_master = self.storage_info[storage.FLAG_KEYS['storage_pool']] | ||||||
|  |             pool_slave = self._get_target_params( | ||||||
|  |                 secondary_backend_id)['san_clustername'] | ||||||
|  |             goal_status = fields.ReplicationStatus.FAILED_OVER | ||||||
|  |         # we should have secondary_backend_id by here. | ||||||
|  |         self.ibm_storage_remote_cli = self._init_xcli(secondary_backend_id) | ||||||
|  |  | ||||||
|  |         # check for split brain in mirrored volumes | ||||||
|  |         self.check_for_splitbrain(volumes, pool_master, pool_slave) | ||||||
|  |         group_specs, msg = self.get_group_specs_by_group_resource(context, | ||||||
|  |                                                                   group) | ||||||
|  |         if group_specs is None: | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.meta['exception'].VolumeBackendAPIException(data=msg) | ||||||
|  |  | ||||||
|  |         failback = (secondary_backend_id == strings.PRIMARY_BACKEND_ID) | ||||||
|  |  | ||||||
|  |         result, details = repl.GroupReplication.failover(group, failback) | ||||||
|  |  | ||||||
|  |         if result: | ||||||
|  |                 status = goal_status | ||||||
|  |                 group_updated['replication_status'] = status | ||||||
|  |         else: | ||||||
|  |             status = 'error' | ||||||
|  |         updates = {'status': status} | ||||||
|  |         if status == 'error': | ||||||
|  |             group_updated['replication_extended_status'] = details | ||||||
|  |         # if replication on cg was successful, then all of the volumes | ||||||
|  |         # have been successfully replicated as well. | ||||||
|  |         for volume in volumes: | ||||||
|  |             volumes_updated.append({ | ||||||
|  |                 'volume_id': volume.id, | ||||||
|  |                 'updates': updates | ||||||
|  |             }) | ||||||
|  |         # replace between active and secondary xcli | ||||||
|  |         self._replace_xcli_to_remote_xcli() | ||||||
|  |  | ||||||
|  |         return group_updated, volumes_updated | ||||||
|  |  | ||||||
|  |     def _replace_xcli_to_remote_xcli(self): | ||||||
|  |         temp_ibm_storage_cli = self.ibm_storage_cli | ||||||
|  |         self.ibm_storage_cli = self.ibm_storage_remote_cli | ||||||
|  |         self.ibm_storage_remote_cli = temp_ibm_storage_cli | ||||||
|  |  | ||||||
|     def _get_replication_target_params(self): |     def _get_replication_target_params(self): | ||||||
|         LOG.debug('_get_replication_target_params.') |         LOG.debug('_get_replication_target_params.') | ||||||
|         targets = self._get_targets() |         if not self.targets: | ||||||
|         if not targets: |  | ||||||
|             msg = _("No targets available for replication") |             msg = _("No targets available for replication") | ||||||
|             LOG.error(msg) |             LOG.error(msg) | ||||||
|             raise self.meta['exception'].VolumeBackendAPIException(data=msg) |             raise self.meta['exception'].VolumeBackendAPIException(data=msg) | ||||||
|         no_of_targets = len(targets) |         no_of_targets = len(self.targets) | ||||||
|         if no_of_targets > 1: |         if no_of_targets > 1: | ||||||
|             msg = _("Too many targets configured. Only one is supported") |             msg = _("Too many targets configured. Only one is supported") | ||||||
|             LOG.error(msg) |             LOG.error(msg) | ||||||
| @@ -591,89 +795,6 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|             raise self.meta['exception'].VolumeBackendAPIException(data=msg) |             raise self.meta['exception'].VolumeBackendAPIException(data=msg) | ||||||
|         return target, params |         return target, params | ||||||
|  |  | ||||||
|     def _replication_create(self, volume, replication_info): |  | ||||||
|         LOG.debug('_replication_create replication_info %(rep)s', |  | ||||||
|                   {'rep': replication_info}) |  | ||||||
|  |  | ||||||
|         target, params = self._get_replication_target_params() |  | ||||||
|         LOG.info('Target %(target)s: %(params)s', |  | ||||||
|                  {'target': target, 'params': six.text_type(params)}) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             pool = params['san_clustername'] |  | ||||||
|         except Exception: |  | ||||||
|             msg = (_("Missing pool information for target '%(target)s'"), |  | ||||||
|                    {'target': target}) |  | ||||||
|             LOG.error(msg) |  | ||||||
|             raise self.meta['exception'].VolumeBackendAPIException(data=msg) |  | ||||||
|  |  | ||||||
|         volume_replication_mgr = volume_recovery_manager.VolumeRecoveryManager( |  | ||||||
|             False, self.ibm_storage_cli) |  | ||||||
|  |  | ||||||
|         self._replication_create_mirror(volume, replication_info, |  | ||||||
|                                         target, pool, volume_replication_mgr) |  | ||||||
|  |  | ||||||
|     def _replication_create_mirror(self, volume, replication_info, |  | ||||||
|                                    target, pool, volume_replication_mgr): |  | ||||||
|         LOG.debug('_replication_create_mirror') |  | ||||||
|         schedule = None |  | ||||||
|         if replication_info['rpo']: |  | ||||||
|             schedule = self._get_schedule_from_rpo(replication_info['rpo']) |  | ||||||
|             if schedule: |  | ||||||
|                 LOG.debug('schedule %(sched)s: for rpo %(rpo)s', |  | ||||||
|                           {'sched': schedule, 'rpo': replication_info['rpo']}) |  | ||||||
|             else: |  | ||||||
|                 LOG.error('Failed to find schedule for rpo %(rpo)s', |  | ||||||
|                           {'rpo': replication_info['rpo']}) |  | ||||||
|                 # will fail in the next step |  | ||||||
|         try: |  | ||||||
|             volume_replication_mgr.create_mirror( |  | ||||||
|                 resource_name=volume['name'], |  | ||||||
|                 target_name=target, |  | ||||||
|                 mirror_type=replication_info['mode'], |  | ||||||
|                 slave_resource_name=volume['name'], |  | ||||||
|                 create_slave='yes', |  | ||||||
|                 remote_pool=pool, |  | ||||||
|                 rpo=replication_info['rpo'], |  | ||||||
|                 schedule=schedule, |  | ||||||
|                 activate_mirror='yes') |  | ||||||
|         # TBD - what exceptions will we get here? |  | ||||||
|         except Exception as e: |  | ||||||
|             details = self._get_code_and_status_or_message(e) |  | ||||||
|             msg = (_("Failed replication for %(vol)s: '%(details)s'"), |  | ||||||
|                    {'vol': volume['name'], 'details': details}) |  | ||||||
|             LOG.error(msg) |  | ||||||
|             raise self.meta['exception'].VolumeBackendAPIException(data=msg) |  | ||||||
|  |  | ||||||
|     def _replication_delete(self, volume, replication_info): |  | ||||||
|         LOG.debug('_replication_delete replication_info %(rep)s', |  | ||||||
|                   {'rep': replication_info}) |  | ||||||
|         targets = self._get_targets() |  | ||||||
|         if not targets: |  | ||||||
|             LOG.debug('No targets defined for replication') |  | ||||||
|  |  | ||||||
|         volume_replication_mgr = volume_recovery_manager.VolumeRecoveryManager( |  | ||||||
|             False, self.ibm_storage_cli) |  | ||||||
|         try: |  | ||||||
|             volume_replication_mgr.deactivate_mirror( |  | ||||||
|                 resource_id=volume['name']) |  | ||||||
|         except Exception as e: |  | ||||||
|             details = self._get_code_and_status_or_message(e) |  | ||||||
|             msg = (_("Failed ending replication for %(vol)s: '%(details)s'"), |  | ||||||
|                    {'vol': volume['name'], 'details': details}) |  | ||||||
|             LOG.error(msg) |  | ||||||
|             raise self.meta['exception'].VolumeBackendAPIException(data=msg) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             volume_replication_mgr.delete_mirror( |  | ||||||
|                 resource_id=volume['name']) |  | ||||||
|         except Exception as e: |  | ||||||
|             details = self._get_code_and_status_or_message(e) |  | ||||||
|             msg = (_("Failed deleting replica for %(vol)s: '%(details)s'"), |  | ||||||
|                    {'vol': volume['name'], 'details': details}) |  | ||||||
|             LOG.error(msg) |  | ||||||
|             raise self.meta['exception'].VolumeBackendAPIException(data=msg) |  | ||||||
|  |  | ||||||
|     def _delete_volume(self, vol_name): |     def _delete_volume(self, vol_name): | ||||||
|         """Deletes a volume on the Storage.""" |         """Deletes a volume on the Storage.""" | ||||||
|         LOG.debug("_delete_volume: %(volume)s", |         LOG.debug("_delete_volume: %(volume)s", | ||||||
| @@ -727,7 +848,8 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         replication_info = self._get_replication_info(specs) |         replication_info = self._get_replication_info(specs) | ||||||
|         if replication_info['enabled']: |         if replication_info['enabled']: | ||||||
|             try: |             try: | ||||||
|                 self._replication_delete(volume, replication_info) |                 repl.VolumeReplication(self).delete_replication( | ||||||
|  |                     volume.name, replication_info) | ||||||
|             except Exception as e: |             except Exception as e: | ||||||
|                 error = self._get_code_and_status_or_message(e) |                 error = self._get_code_and_status_or_message(e) | ||||||
|                 LOG.error(DELETE_VOLUME_BASE_ERROR, |                 LOG.error(DELETE_VOLUME_BASE_ERROR, | ||||||
| @@ -1221,70 +1343,6 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|                 potential_split_brain.append(name) |                 potential_split_brain.append(name) | ||||||
|         return potential_split_brain |         return potential_split_brain | ||||||
|  |  | ||||||
|     def _failover_vol(self, volume, volume_replication_mgr, |  | ||||||
|                       failover_volume_replication_mgr, |  | ||||||
|                       replication_info, failback): |  | ||||||
|         """Failover a single volume. |  | ||||||
|  |  | ||||||
|         Attempts to failover a single volume |  | ||||||
|         Sequence: |  | ||||||
|         1. attempt to switch roles from master |  | ||||||
|         2. attempt to change role to master on slave |  | ||||||
|  |  | ||||||
|         returns (success, failure_reason) |  | ||||||
|         """ |  | ||||||
|         LOG.debug("_failover_vol %(vol)s", {'vol': volume['name']}) |  | ||||||
|  |  | ||||||
|         # check if mirror is defined and active |  | ||||||
|         try: |  | ||||||
|             LOG.debug('Check if mirroring is active on %(vol)s.', |  | ||||||
|                       {'vol': volume['name']}) |  | ||||||
|             active = volume_replication_mgr.is_mirror_active( |  | ||||||
|                 resource_id=volume['name']) |  | ||||||
|         except Exception: |  | ||||||
|             active = False |  | ||||||
|         state = 'active' if active else 'inactive' |  | ||||||
|         LOG.debug('Mirroring is %(state)s', {'state': state}) |  | ||||||
|  |  | ||||||
|         # In case of failback, mirroring must be active |  | ||||||
|         # In case of failover we attempt to move in any condition |  | ||||||
|         if failback and not active: |  | ||||||
|             msg = ("Volume %(vol)s: no active mirroring and can not " |  | ||||||
|                    "failback", |  | ||||||
|                    {'vol': volume['name']}) |  | ||||||
|             LOG.error(msg) |  | ||||||
|             return False, msg |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             volume_replication_mgr.switch_roles(resource_id=volume['name']) |  | ||||||
|             return True, None |  | ||||||
|         except Exception as e: |  | ||||||
|             # failed attempt to switch_roles from the master |  | ||||||
|             details = self._get_code_and_status_or_message(e) |  | ||||||
|             LOG.warning('Failed to perform switch_roles on' |  | ||||||
|                         ' %(vol)s: %(err)s. ' |  | ||||||
|                         'Continue to change_role', |  | ||||||
|                         {'vol': volume['name'], 'err': details}) |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             # this is the ugly stage we come to brute force |  | ||||||
|             LOG.warning('Attempt to change_role to master') |  | ||||||
|             failover_volume_replication_mgr.failover_by_id( |  | ||||||
|                 resource_id=volume['name']) |  | ||||||
|             return True, None |  | ||||||
|         except m_errors.NoMirrorDefinedError as e: |  | ||||||
|             details = self._get_code_and_status_or_message(e) |  | ||||||
|             msg = ("Volume %(vol)s no replication defined: %(err)s" % |  | ||||||
|                    {'vol': volume['name'], 'err': details}) |  | ||||||
|             LOG.error(msg) |  | ||||||
|             return False, msg |  | ||||||
|         except Exception as e: |  | ||||||
|             details = self._get_code_and_status_or_message(e) |  | ||||||
|             msg = ('Volume %(vol)s change_role failed: %(err)s' % |  | ||||||
|                    {'vol': volume['name'], 'err': details}) |  | ||||||
|             LOG.error(msg) |  | ||||||
|             return False, msg |  | ||||||
|  |  | ||||||
|     @proxy._trace_time |     @proxy._trace_time | ||||||
|     def failover_host(self, context, volumes, secondary_id, groups=None): |     def failover_host(self, context, volumes, secondary_id, groups=None): | ||||||
|         """Failover a full backend. |         """Failover a full backend. | ||||||
| @@ -1316,14 +1374,7 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|                 LOG.info("Already failed over. No need to failover again.") |                 LOG.info("Already failed over. No need to failover again.") | ||||||
|                 return self.active_backend_id, volume_update_list, [] |                 return self.active_backend_id, volume_update_list, [] | ||||||
|             # case: need to select a target |             # case: need to select a target | ||||||
|             if secondary_id is None: |             secondary_id = self.get_secondary_backend_id(secondary_id) | ||||||
|                 secondary_id = self._get_target() |  | ||||||
|             # still no targets.. |  | ||||||
|             if secondary_id is None: |  | ||||||
|                 msg = _("No targets defined. Can't perform failover") |  | ||||||
|                 LOG.error(msg) |  | ||||||
|                 raise self.meta['exception'].VolumeBackendAPIException( |  | ||||||
|                     data=msg) |  | ||||||
|             pool_master = self.storage_info[storage.FLAG_KEYS['storage_pool']] |             pool_master = self.storage_info[storage.FLAG_KEYS['storage_pool']] | ||||||
|             try: |             try: | ||||||
|                 pool_slave = self._get_target_params( |                 pool_slave = self._get_target_params( | ||||||
| @@ -1340,45 +1391,21 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         #  calling _init_xcli with secondary_id |         #  calling _init_xcli with secondary_id | ||||||
|         self.ibm_storage_remote_cli = self._init_xcli(secondary_id) |         self.ibm_storage_remote_cli = self._init_xcli(secondary_id) | ||||||
|  |  | ||||||
|         # Create volume manager for both master and remote |  | ||||||
|         volume_replication_mgr = volume_recovery_manager.VolumeRecoveryManager( |  | ||||||
|             False, self.ibm_storage_cli) |  | ||||||
|         failover_volume_replication_mgr = ( |  | ||||||
|             volume_recovery_manager.VolumeRecoveryManager( |  | ||||||
|                 True, self.ibm_storage_remote_cli)) |  | ||||||
|  |  | ||||||
|         # get replication_info for all volumes at once |         # get replication_info for all volumes at once | ||||||
|         if len(volumes): |         if len(volumes): | ||||||
|             # check for split brain situations |             # check for split brain situations | ||||||
|             # check for files that are available on both volumes |             # check for files that are available on both volumes | ||||||
|             # and are not in an active mirroring relation |             # and are not in an active mirroring relation | ||||||
|             split_brain = self._potential_split_brain( |             self.check_for_splitbrain(volumes, pool_master, pool_slave) | ||||||
|                 self.ibm_storage_cli, |  | ||||||
|                 self.ibm_storage_remote_cli, |  | ||||||
|                 volumes, pool_master, |  | ||||||
|                 pool_slave) |  | ||||||
|             if len(split_brain): |  | ||||||
|                 # if such a situation exists stop and raise an exception! |  | ||||||
|                 msg = (_("A potential split brain condition has been found " |  | ||||||
|                          "with the following volumes: \n'%(volumes)s.'"), |  | ||||||
|                        {'volumes': split_brain}) |  | ||||||
|                 LOG.error(msg) |  | ||||||
|                 raise self.meta['exception'].VolumeBackendAPIException( |  | ||||||
|                     data=msg) |  | ||||||
|             specs = self._get_extra_specs( |  | ||||||
|                 volumes[0].get('volume_type_id', None)) |  | ||||||
|             replication_info = self._get_replication_info(specs) |  | ||||||
|  |  | ||||||
|         # loop over volumes and attempt failover |         # loop over volumes and attempt failover | ||||||
|         for volume in volumes: |         for volume in volumes: | ||||||
|             LOG.debug("Attempting to failover '%(vol)s'", |             LOG.debug("Attempting to failover '%(vol)s'", | ||||||
|                       {'vol': volume['name']}) |                       {'vol': volume['name']}) | ||||||
|             result, details = self._failover_vol( |  | ||||||
|                 volume, |             result, details = repl.VolumeReplication(self).failover( | ||||||
|                 volume_replication_mgr, |                 volume, failback=(secondary_id == strings.PRIMARY_BACKEND_ID)) | ||||||
|                 failover_volume_replication_mgr, |  | ||||||
|                 replication_info, |  | ||||||
|                 failback=(secondary_id == strings.PRIMARY_BACKEND_ID)) |  | ||||||
|             if result: |             if result: | ||||||
|                 status = goal_status |                 status = goal_status | ||||||
|             else: |             else: | ||||||
| @@ -1393,9 +1420,7 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|             }) |             }) | ||||||
|  |  | ||||||
|         # set active xcli to secondary xcli |         # set active xcli to secondary xcli | ||||||
|         temp_ibm_storage_cli = self.ibm_storage_cli |         self._replace_xcli_to_remote_xcli() | ||||||
|         self.ibm_storage_cli = self.ibm_storage_remote_cli |  | ||||||
|         self.ibm_storage_remote_cli = temp_ibm_storage_cli |  | ||||||
|         # set active backend id to secondary id |         # set active backend id to secondary id | ||||||
|         self.active_backend_id = secondary_id |         self.active_backend_id = secondary_id | ||||||
|  |  | ||||||
| @@ -1500,6 +1525,8 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         self.meta['stat']["driver_version"] = self.full_version |         self.meta['stat']["driver_version"] = self.full_version | ||||||
|         self.meta['stat']["storage_protocol"] = connection_type |         self.meta['stat']["storage_protocol"] = connection_type | ||||||
|         self.meta['stat']['multiattach'] = False |         self.meta['stat']['multiattach'] = False | ||||||
|  |         self.meta['stat']['group_replication_enabled'] = True | ||||||
|  |         self.meta['stat']['consistent_group_replication_enabled'] = True | ||||||
|         self.meta['stat']['QoS_support'] = ( |         self.meta['stat']['QoS_support'] = ( | ||||||
|             self._check_storage_version_for_qos_support()) |             self._check_storage_version_for_qos_support()) | ||||||
|  |  | ||||||
| @@ -1538,15 +1565,14 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         self.meta['stat']['thin_provision'] = ('True' if soft_size > hard_size |         self.meta['stat']['thin_provision'] = ('True' if soft_size > hard_size | ||||||
|                                                else 'False') |                                                else 'False') | ||||||
|  |  | ||||||
|         targets = self._get_targets() |         if self.targets: | ||||||
|         if targets: |  | ||||||
|             self.meta['stat']['replication_enabled'] = True |             self.meta['stat']['replication_enabled'] = True | ||||||
|             # TBD - replication_type should be according to type |  | ||||||
|             self.meta['stat']['replication_type'] = [SYNC, ASYNC] |             self.meta['stat']['replication_type'] = [SYNC, ASYNC] | ||||||
|             self.meta['stat']['rpo'] = self._get_supported_rpo() |             self.meta['stat']['rpo'] = repl.Replication.get_supported_rpo() | ||||||
|             self.meta['stat']['replication_count'] = len(targets) |             self.meta['stat']['replication_count'] = len(self.targets) | ||||||
|             self.meta['stat']['replication_targets'] = [target for target in |             self.meta['stat']['replication_targets'] = [target for target in | ||||||
|                                                         six.iterkeys(targets)] |                                                         six.iterkeys( | ||||||
|  |                                                             self.targets)] | ||||||
|  |  | ||||||
|         self.meta['stat']['timestamp'] = datetime.datetime.utcnow() |         self.meta['stat']['timestamp'] = datetime.datetime.utcnow() | ||||||
|  |  | ||||||
| @@ -1679,18 +1705,6 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|     def create_group(self, context, group): |     def create_group(self, context, group): | ||||||
|         """Creates a group.""" |         """Creates a group.""" | ||||||
|  |  | ||||||
|         for volume_type in group.volume_types: |  | ||||||
|             replication_info = self._get_replication_info( |  | ||||||
|                 volume_type.extra_specs) |  | ||||||
|  |  | ||||||
|             if replication_info.get('enabled'): |  | ||||||
|                 # An unsupported illegal configuration |  | ||||||
|                 msg = _("Unable to create group: create group with " |  | ||||||
|                         "replication volume type is not supported") |  | ||||||
|                 LOG.error(msg) |  | ||||||
|                 raise self.meta['exception'].VolumeBackendAPIException( |  | ||||||
|                     data=msg) |  | ||||||
|  |  | ||||||
|         if utils.is_group_a_cg_snapshot_type(group): |         if utils.is_group_a_cg_snapshot_type(group): | ||||||
|             cgname = self._cg_name_from_group(group) |             cgname = self._cg_name_from_group(group) | ||||||
|             return self._create_consistencygroup(context, cgname) |             return self._create_consistencygroup(context, cgname) | ||||||
| @@ -1726,6 +1740,35 @@ class XIVProxy(proxy.IBMStorageProxy): | |||||||
|         model_update = {'status': fields.GroupStatus.AVAILABLE} |         model_update = {'status': fields.GroupStatus.AVAILABLE} | ||||||
|         return model_update |         return model_update | ||||||
|  |  | ||||||
|  |     def _create_consistencygroup_on_remote(self, context, cgname): | ||||||
|  |         """Creates a consistency group on secondary machine. | ||||||
|  |  | ||||||
|  |         Return group available even if it already exists (for replication) | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         LOG.info("Creating consistency group %(name)s on secondary.", | ||||||
|  |                  {'name': cgname}) | ||||||
|  |  | ||||||
|  |         # call remote XCLI | ||||||
|  |         try: | ||||||
|  |             self._call_remote_xiv_xcli( | ||||||
|  |                 "cg_create", cg=cgname, | ||||||
|  |                 pool=self.storage_info[ | ||||||
|  |                     storage.FLAG_KEYS['storage_pool']]).as_list | ||||||
|  |         except errors.CgNameExistsError: | ||||||
|  |             model_update = {'status': fields.GroupStatus.AVAILABLE} | ||||||
|  |         except errors.CgLimitReachedError: | ||||||
|  |             error = _("Maximum number of consistency groups reached") | ||||||
|  |             LOG.error(error) | ||||||
|  |             raise self._get_exception()(error) | ||||||
|  |         except errors.XCLIError as e: | ||||||
|  |             error = (_("Fatal error in cg_create on remote: %(details)s") % | ||||||
|  |                      {'details': self._get_code_and_status_or_message(e)}) | ||||||
|  |             LOG.error(error) | ||||||
|  |             raise self._get_exception()(error) | ||||||
|  |         model_update = {'status': fields.GroupStatus.AVAILABLE} | ||||||
|  |         return model_update | ||||||
|  |  | ||||||
|     def _silent_cleanup_consistencygroup_from_src(self, context, group, |     def _silent_cleanup_consistencygroup_from_src(self, context, group, | ||||||
|                                                   volumes, cgname): |                                                   volumes, cgname): | ||||||
|         """Silent cleanup of volumes from CG. |         """Silent cleanup of volumes from CG. | ||||||
|   | |||||||
							
								
								
									
										342
									
								
								cinder/volume/drivers/ibm/ibm_storage/xiv_replication.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										342
									
								
								cinder/volume/drivers/ibm/ibm_storage/xiv_replication.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,342 @@ | |||||||
|  | #  Copyright (c) 2017 IBM Corporation | ||||||
|  | #  All Rights Reserved. | ||||||
|  | # | ||||||
|  | #    Licensed under the Apache License, Version 2.0 (the "License"); you may | ||||||
|  | #    not use this file except in compliance with the License. You may obtain | ||||||
|  | #    a copy of the License at | ||||||
|  | # | ||||||
|  | #         http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  | # | ||||||
|  | #    Unless required by applicable law or agreed to in writing, software | ||||||
|  | #    distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | ||||||
|  | #    WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||||||
|  | #    License for the specific language governing permissions and limitations | ||||||
|  | #    under the License. | ||||||
|  | # | ||||||
|  | import six | ||||||
|  |  | ||||||
|  | from oslo_log import log as logging | ||||||
|  | from oslo_utils import importutils | ||||||
|  |  | ||||||
|  | pyxcli = importutils.try_import("pyxcli") | ||||||
|  | if pyxcli: | ||||||
|  |     from pyxcli import errors | ||||||
|  |     from pyxcli.mirroring import cg_recovery_manager | ||||||
|  |     from pyxcli.mirroring import errors as m_errors | ||||||
|  |     from pyxcli.mirroring import volume_recovery_manager | ||||||
|  |  | ||||||
|  | from cinder.i18n import _ | ||||||
|  | from cinder.volume.drivers.ibm.ibm_storage import strings | ||||||
|  |  | ||||||
|  | SYNC = 'sync' | ||||||
|  | ASYNC = 'async' | ||||||
|  |  | ||||||
|  | LOG = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Rate(object): | ||||||
|  |  | ||||||
|  |     def __init__(self, rpo, schedule): | ||||||
|  |         self.rpo = rpo | ||||||
|  |         self.schedule = schedule | ||||||
|  |         self.schedule_name = self._schedule_name_from_schedule(self.schedule) | ||||||
|  |  | ||||||
|  |     def _schedule_name_from_schedule(self, schedule): | ||||||
|  |         if schedule == '00:00:20': | ||||||
|  |             return 'min_interval' | ||||||
|  |         return ("cinder_%(sched)s" % | ||||||
|  |                 {'sched': schedule.replace(':', '_')}) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Replication(object): | ||||||
|  |  | ||||||
|  |     async_rates = ( | ||||||
|  |         Rate(rpo=120, schedule='00:01:00'), | ||||||
|  |         Rate(rpo=300, schedule='00:02:00'), | ||||||
|  |         Rate(rpo=600, schedule='00:05:00'), | ||||||
|  |         Rate(rpo=1200, schedule='00:10:00'), | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     def __init__(self, proxy): | ||||||
|  |         self.proxy = proxy | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def get_schedule_from_rpo(rpo): | ||||||
|  |         schedule = [rate for rate in Replication.async_rates | ||||||
|  |                     if rate.rpo == rpo][0].schedule_name | ||||||
|  |         if schedule: | ||||||
|  |             LOG.debug('schedule %(sched)s: for rpo %(rpo)s', | ||||||
|  |                       {'sched': schedule, 'rpo': rpo}) | ||||||
|  |         else: | ||||||
|  |             LOG.error('Failed to find schedule for rpo %(rpo)s', | ||||||
|  |                       {'rpo': rpo}) | ||||||
|  |         return schedule | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def get_supported_rpo(): | ||||||
|  |         return [rate.rpo for rate in Replication.async_rates] | ||||||
|  |  | ||||||
|  |     def get_recovery_mgr(self): | ||||||
|  |         # Recovery manager is set in derived classes | ||||||
|  |         raise NotImplementedError | ||||||
|  |  | ||||||
|  |     def get_remote_recovery_mgr(self): | ||||||
|  |         # Recovery manager is set in derived classes | ||||||
|  |         raise NotImplementedError | ||||||
|  |  | ||||||
|  |     def replication_create_mirror(self, resource, replication_info, | ||||||
|  |                                   target, pool): | ||||||
|  |         raise NotImplementedError | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def extract_replication_info_from_specs(specs): | ||||||
|  |         info = {'enabled': False, 'mode': None, 'rpo': 0} | ||||||
|  |         msg = "" | ||||||
|  |         if specs: | ||||||
|  |             LOG.debug('extract_replication_info_from_specs: specs %(specs)s', | ||||||
|  |                       {'specs': specs}) | ||||||
|  |  | ||||||
|  |             info['enabled'] = ( | ||||||
|  |                 specs.get('replication_enabled', '').upper() in | ||||||
|  |                 (u'TRUE', strings.METADATA_IS_TRUE) or | ||||||
|  |                 specs.get('group_replication_enabled', '').upper() in | ||||||
|  |                 (u'TRUE', strings.METADATA_IS_TRUE)) | ||||||
|  |  | ||||||
|  |             replication_type = specs.get('replication_type', SYNC).lower() | ||||||
|  |             if replication_type in (u'sync', u'<is> sync'): | ||||||
|  |                 info['mode'] = SYNC | ||||||
|  |             elif replication_type in (u'async', u'<is> async'): | ||||||
|  |                 info['mode'] = ASYNC | ||||||
|  |             else: | ||||||
|  |                 msg = (_("Unsupported replication mode %(mode)s") | ||||||
|  |                        % {'mode': replication_type}) | ||||||
|  |                 return None, msg | ||||||
|  |             info['rpo'] = int(specs.get('rpo', u'<is> 0')[5:]) | ||||||
|  |             supported_rpos = Replication.get_supported_rpo() | ||||||
|  |             if info['rpo'] and info['rpo'] not in supported_rpos: | ||||||
|  |                 msg = (_("Unsupported replication RPO %(rpo)s"), | ||||||
|  |                        {'rpo': info['rpo']}) | ||||||
|  |                 return None, msg | ||||||
|  |  | ||||||
|  |             LOG.debug('extract_replication_info_from_specs: info %(info)s', | ||||||
|  |                       {'info': info}) | ||||||
|  |         return info, msg | ||||||
|  |  | ||||||
|  |     def failover(self, resource, failback): | ||||||
|  |         raise NotImplementedError | ||||||
|  |  | ||||||
|  |     def create_replication(self, resource_name, replication_info): | ||||||
|  |         LOG.debug('Replication::create_replication replication_info %(rep)s', | ||||||
|  |                   {'rep': replication_info}) | ||||||
|  |  | ||||||
|  |         target, params = self.proxy._get_replication_target_params() | ||||||
|  |         LOG.info('Target %(target)s: %(params)s', | ||||||
|  |                  {'target': target, 'params': six.text_type(params)}) | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             pool = params['san_clustername'] | ||||||
|  |         except Exception: | ||||||
|  |             msg = (_("Missing pool information for target '%(target)s'") % | ||||||
|  |                    {'target': target}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.proxy.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                 data=msg) | ||||||
|  |  | ||||||
|  |         self.replication_create_mirror(resource_name, replication_info, | ||||||
|  |                                        target, pool) | ||||||
|  |  | ||||||
|  |     def delete_replication(self, resource_name, replication_info): | ||||||
|  |         LOG.debug('Replication::delete_replication replication_info %(rep)s', | ||||||
|  |                   {'rep': replication_info}) | ||||||
|  |  | ||||||
|  |         recovery_mgr = self.get_recovery_mgr() | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             recovery_mgr.deactivate_mirror(resource_id=resource_name) | ||||||
|  |         except Exception as e: | ||||||
|  |             details = self.proxy._get_code_and_status_or_message(e) | ||||||
|  |             msg = (_("Failed ending replication for %(resource)s: " | ||||||
|  |                      "'%(details)s'") % {'resource': resource_name, | ||||||
|  |                                          'details': details}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.proxy.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                 data=msg) | ||||||
|  |         try: | ||||||
|  |             recovery_mgr.delete_mirror(resource_id=resource_name) | ||||||
|  |         except Exception as e: | ||||||
|  |             details = self.proxy._get_code_and_status_or_message(e) | ||||||
|  |             msg = (_("Failed deleting replica for %(resource)s: " | ||||||
|  |                      "'%(details)s'") % {'resource': resource_name, | ||||||
|  |                                          'details': details}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.proxy.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                 data=msg) | ||||||
|  |  | ||||||
|  |     def _failover_resource(self, resource, recovery_mgr, failover_rep_mgr, | ||||||
|  |                            rep_type, failback): | ||||||
|  |         # check if mirror is defined and active | ||||||
|  |         LOG.debug('Check if mirroring is active on %(res)s', | ||||||
|  |                   {'res': resource['name']}) | ||||||
|  |         try: | ||||||
|  |             active = recovery_mgr.is_mirror_active( | ||||||
|  |                 resource_id=resource['name']) | ||||||
|  |         except Exception: | ||||||
|  |             active = False | ||||||
|  |         state = 'active' if active else 'inactive' | ||||||
|  |         LOG.debug('Mirroring is %(state)s', {'state': state}) | ||||||
|  |  | ||||||
|  |         # In case of failback, mirroring must be active | ||||||
|  |         # In case of failover we attempt to move in any condition | ||||||
|  |         if failback and not active: | ||||||
|  |             msg = ("%(rep_type)s %(res)s: no active mirroring and can not " | ||||||
|  |                    "failback" % {'rep_type': rep_type, | ||||||
|  |                                  'res': resource['name']}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             return False, msg | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             recovery_mgr.switch_roles(resource_id=resource['name']) | ||||||
|  |             return True, None | ||||||
|  |         except Exception as e: | ||||||
|  |             # failed attempt to switch_roles from the master | ||||||
|  |             details = self.proxy._get_code_and_status_or_message(e) | ||||||
|  |             LOG.warning('Failed to perform switch_roles on' | ||||||
|  |                         ' %(res)s: %(err)s. ' | ||||||
|  |                         'Continue to change_role', | ||||||
|  |                         {'res': resource['name'], 'err': details}) | ||||||
|  |         try: | ||||||
|  |             # this is the ugly stage we come to brute force | ||||||
|  |             if failback: | ||||||
|  |                 role = 'Slave' | ||||||
|  |             else: | ||||||
|  |                 role = 'Master' | ||||||
|  |             LOG.warning('Attempt to change_role to %(role)s', {'role': role}) | ||||||
|  |             failover_rep_mgr.change_role(resource_id=resource['name'], | ||||||
|  |                                          new_role=role) | ||||||
|  |             return True, None | ||||||
|  |         except m_errors.NoMirrorDefinedError as e: | ||||||
|  |             details = self.proxy._get_code_and_status_or_message(e) | ||||||
|  |             msg = ("%(rep_type)s %(res)s no replication defined: %(err)s" % | ||||||
|  |                    {'rep_type': rep_type, 'res': resource['name'], | ||||||
|  |                     'err': details}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             return False, msg | ||||||
|  |         except Exception as e: | ||||||
|  |             details = self.proxy._get_code_and_status_or_message(e) | ||||||
|  |             msg = ('%(rep_type)s %(res)s change_role failed: %(err)s' % | ||||||
|  |                    {'rep_type': rep_type, 'res': resource['name'], | ||||||
|  |                     'err': details}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             return False, msg | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class VolumeReplication(Replication): | ||||||
|  |  | ||||||
|  |     def __init__(self, proxy): | ||||||
|  |         super(VolumeReplication, self).__init__(proxy) | ||||||
|  |  | ||||||
|  |     def get_recovery_mgr(self): | ||||||
|  |         return volume_recovery_manager.VolumeRecoveryManager( | ||||||
|  |             False, self.proxy.ibm_storage_cli) | ||||||
|  |  | ||||||
|  |     def get_remote_recovery_mgr(self): | ||||||
|  |         return volume_recovery_manager.VolumeRecoveryManager( | ||||||
|  |             True, self.proxy.ibm_storage_remote_cli) | ||||||
|  |  | ||||||
|  |     def replication_create_mirror(self, resource_name, replication_info, | ||||||
|  |                                   target, pool): | ||||||
|  |         LOG.debug('VolumeReplication::replication_create_mirror') | ||||||
|  |  | ||||||
|  |         schedule = None | ||||||
|  |         if replication_info['rpo']: | ||||||
|  |             schedule = Replication.get_schedule_from_rpo( | ||||||
|  |                 replication_info['rpo']) | ||||||
|  |         try: | ||||||
|  |             recovery_mgr = self.get_recovery_mgr() | ||||||
|  |             recovery_mgr.create_mirror( | ||||||
|  |                 resource_name=resource_name, | ||||||
|  |                 target_name=target, | ||||||
|  |                 mirror_type=replication_info['mode'], | ||||||
|  |                 slave_resource_name=resource_name, | ||||||
|  |                 create_slave='yes', | ||||||
|  |                 remote_pool=pool, | ||||||
|  |                 rpo=replication_info['rpo'], | ||||||
|  |                 schedule=schedule, | ||||||
|  |                 activate_mirror='yes') | ||||||
|  |         except errors.VolumeMasterError: | ||||||
|  |             LOG.debug('Volume %(vol)s has been already mirrored', | ||||||
|  |                       {'vol': resource_name}) | ||||||
|  |         except Exception as e: | ||||||
|  |             details = self.proxy._get_code_and_status_or_message(e) | ||||||
|  |             msg = (_("Failed replication for %(resource)s: '%(details)s'") % | ||||||
|  |                    {'resource': resource_name, 'details': details}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.proxy.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                 data=msg) | ||||||
|  |  | ||||||
|  |     def failover(self, resource, failback): | ||||||
|  |         """Failover a single volume. | ||||||
|  |  | ||||||
|  |         Attempts to failover a single volume | ||||||
|  |         Sequence: | ||||||
|  |         1. attempt to switch roles from master | ||||||
|  |         2. attempt to change role to master on secondary | ||||||
|  |  | ||||||
|  |         returns (success, failure_reason) | ||||||
|  |         """ | ||||||
|  |         LOG.debug("VolumeReplication::failover %(vol)s", | ||||||
|  |                   {'vol': resource['name']}) | ||||||
|  |  | ||||||
|  |         recovery_mgr = self.get_recovery_mgr() | ||||||
|  |         remote_recovery_mgr = self.get_remote_recovery_mgr() | ||||||
|  |         return self._failover_resource(resource, recovery_mgr, | ||||||
|  |                                        remote_recovery_mgr, 'vol', failback) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GroupReplication(Replication): | ||||||
|  |  | ||||||
|  |     def __init__(self, proxy): | ||||||
|  |         super(GroupReplication, self).__init__(proxy) | ||||||
|  |  | ||||||
|  |     def get_recovery_mgr(self): | ||||||
|  |         return cg_recovery_manager.CGRecoveryManager( | ||||||
|  |             False, self.proxy.ibm_storage_cli) | ||||||
|  |  | ||||||
|  |     def get_remote_recovery_mgr(self): | ||||||
|  |         return volume_recovery_manager.CGRecoveryManager( | ||||||
|  |             True, self.proxy.ibm_storage_remote_cli) | ||||||
|  |  | ||||||
|  |     def replication_create_mirror(self, resource_name, replication_info, | ||||||
|  |                                   target, pool): | ||||||
|  |         LOG.debug('GroupReplication::replication_create_mirror') | ||||||
|  |         schedule = None | ||||||
|  |         if replication_info['rpo']: | ||||||
|  |             schedule = Replication.get_schedule_from_rpo( | ||||||
|  |                 replication_info['rpo']) | ||||||
|  |         try: | ||||||
|  |             recovery_mgr = self.get_recovery_mgr() | ||||||
|  |             recovery_mgr.create_mirror( | ||||||
|  |                 resource_name=resource_name, | ||||||
|  |                 target_name=target, | ||||||
|  |                 mirror_type=replication_info['mode'], | ||||||
|  |                 slave_resource_name=resource_name, | ||||||
|  |                 rpo=replication_info['rpo'], | ||||||
|  |                 schedule=schedule, | ||||||
|  |                 activate_mirror='yes') | ||||||
|  |         except Exception as e: | ||||||
|  |             details = self.proxy._get_code_and_status_or_message(e) | ||||||
|  |             msg = (_("Failed replication for %(resource)s: '%(details)s'"), | ||||||
|  |                    {'resource': resource_name, 'details': details}) | ||||||
|  |             LOG.error(msg) | ||||||
|  |             raise self.proxy.meta['exception'].VolumeBackendAPIException( | ||||||
|  |                 data=msg) | ||||||
|  |  | ||||||
|  |     def failover(self, resource, failback): | ||||||
|  |         LOG.debug("GroupReplication::failover %(cg)s", | ||||||
|  |                   {'cg': resource['name']}) | ||||||
|  |  | ||||||
|  |         recovery_mgr = self.get_recovery_mgr() | ||||||
|  |         remote_recovery_mgr = self.get_remote_recovery_mgr() | ||||||
|  |  | ||||||
|  |         return self._failover_resource(resource, recovery_mgr, | ||||||
|  |                                        remote_recovery_mgr, 'cg', failback) | ||||||
| @@ -0,0 +1,4 @@ | |||||||
|  | --- | ||||||
|  | features: | ||||||
|  |   - | | ||||||
|  |     Add consistency group replication support in XIV\A9000 Cinder driver. | ||||||
		Reference in New Issue
	
	Block a user