From 0a47ab83b5a5c12b31cc5f426000a033788c872c Mon Sep 17 00:00:00 2001 From: TzurEliyahu Date: Thu, 2 Feb 2017 15:05:05 +0200 Subject: [PATCH] xiv create vol from replicated source fails The ibm_storage driver fails to clone a volume from a replicated source. It creates the volume, defines replication for it, and when it tries to copy the contents of the source volume, it fails. The logic should be reversed - after create, we should copy the contents of the source and then set the replication Change-Id: Ic6d093bfc3c03807211dd47c5eb4428127e28821 closes-bug: 1661223 --- .../unit/volume/drivers/ibm/test_xiv_proxy.py | 52 +++++++++++++++++++ .../drivers/ibm/ibm_storage/xiv_proxy.py | 40 ++++++++------ 2 files changed, 76 insertions(+), 16 deletions(-) diff --git a/cinder/tests/unit/volume/drivers/ibm/test_xiv_proxy.py b/cinder/tests/unit/volume/drivers/ibm/test_xiv_proxy.py index 0bf894685db..7fd8e114cda 100644 --- a/cinder/tests/unit/volume/drivers/ibm/test_xiv_proxy.py +++ b/cinder/tests/unit/volume/drivers/ibm/test_xiv_proxy.py @@ -1758,3 +1758,55 @@ class XIVProxyTest(unittest.TestCase): # check no assertion occurs p._silent_delete_volume(TEST_VOLUME) + + def test_create_cloned_volume_calls_vol_create_and_copy(self): + """test create_cloned_volume + + check if calls the appropriate xiv_backend functions + are being called + """ + driver = mock.MagicMock() + driver.VERSION = "VERSION" + + p = self.proxy( + self.default_storage_info, + mock.MagicMock(), + test_mock.cinder.exception, + driver) + + vol_src = {'name': 'bla', 'size': 17} + vol_trg = {'name': 'bla', 'size': 17} + p.ibm_storage_cli = mock.MagicMock() + p._cg_name_from_volume = mock.MagicMock(return_value="cg") + + p.create_cloned_volume(vol_trg, vol_src) + p._create_volume = test_mock.MagicMock() + + p.ibm_storage_cli.cmd.vol_create.assert_called_once_with( + pool='WTF32', + size_blocks=storage.gigabytes_to_blocks(17), + vol=vol_trg['name']) + + p.ibm_storage_cli.cmd.vol_copy.assert_called_once_with( + vol_src=vol_src['name'], + vol_trg=vol_trg['name']) + + def test_handle_created_vol_properties_returns_vol_update(self): + """test handle_created_vol_props + + returns replication enables if replication info is True + """ + driver = mock.MagicMock() + driver.VERSION = "VERSION" + + p = self.proxy( + self.default_storage_info, + mock.MagicMock(), + test_mock.cinder.exception, + driver) + + p._replication_create = test_mock.MagicMock(return_value=None) + ret_val = p.handle_created_vol_properties( + None, {'enabled': True}, {'name': 'bla'}) + + self.assertEqual(ret_val, {'replication_status': 'enabled'}) diff --git a/cinder/volume/drivers/ibm/ibm_storage/xiv_proxy.py b/cinder/volume/drivers/ibm/ibm_storage/xiv_proxy.py index 17f4088c27b..bdfb7701030 100644 --- a/cinder/volume/drivers/ibm/ibm_storage/xiv_proxy.py +++ b/cinder/volume/drivers/ibm/ibm_storage/xiv_proxy.py @@ -461,9 +461,13 @@ class XIVProxy(proxy.IBMStorageProxy): LOG.error(msg) raise self.meta['exception'].VolumeBackendAPIException(data=msg) - volume_update = {} self._create_volume(volume) + return self.handle_created_vol_properties(cg, + replication_info, + volume) + def handle_created_vol_properties(self, cg, replication_info, volume): + volume_update = {} if cg: volume_update['consistencygroup_id'] = ( volume.get('consistencygroup_id', None)) @@ -1513,6 +1517,10 @@ class XIVProxy(proxy.IBMStorageProxy): @proxy._trace_time def create_cloned_volume(self, volume, src_vref): """Create cloned volume.""" + cg = self._cg_name_from_volume(volume) + # read replication information + specs = self._get_extra_specs(volume.get('volume_type_id', None)) + replication_info = self._get_replication_info(specs) # TODO(alonma): Refactor to use more common code src_vref_size = float(src_vref['size']) @@ -1524,7 +1532,7 @@ class XIVProxy(proxy.IBMStorageProxy): LOG.error(error) raise self._get_exception()(error) - self.create_volume(volume) + self._create_volume(volume) try: self._call_xiv_xcli( "vol_copy", @@ -1542,20 +1550,20 @@ class XIVProxy(proxy.IBMStorageProxy): # A side effect of vol_copy is the resizing of the destination volume # to the size of the source volume. If the size is different we need # to get it back to the desired size - if src_vref_size == volume_size: - return - size = storage.gigabytes_to_blocks(volume_size) - try: - self._call_xiv_xcli( - "vol_resize", - vol=volume['name'], - size_blocks=size) - except errors.XCLIError as e: - error = (_("Fatal error in vol_resize: %(details)s"), - {'details': self._get_code_and_status_or_message(e)}) - LOG.error(error) - self._silent_delete_volume(volume=volume) - raise self._get_exception()(error) + if src_vref_size != volume_size: + size = storage.gigabytes_to_blocks(volume_size) + try: + self._call_xiv_xcli( + "vol_resize", + vol=volume['name'], + size_blocks=size) + except errors.XCLIError as e: + error = (_("Fatal error in vol_resize: %(details)s"), + {'details': self._get_code_and_status_or_message(e)}) + LOG.error(error) + self._silent_delete_volume(volume=volume) + raise self._get_exception()(error) + self.handle_created_vol_properties(cg, replication_info, volume) @proxy._trace_time def volume_exists(self, volume):