Dell SC: Use Live Volume for replication
Adding support for Dell SC live volume in replication. If an extra spec is set the driver will attempt to create live volumes instead of normal replications. Added to the test coverage. Minor refactor to failback. DocImpact Change-Id: Id0056ec9b9a42005b1f7cb2d8cd5aee208d8015c
This commit is contained in:
parent
aaf820f429
commit
fecbf75edc
@ -176,7 +176,7 @@ class DellSCSanFCDriverTestCase(test.TestCase):
|
|||||||
'find_server',
|
'find_server',
|
||||||
return_value=None)
|
return_value=None)
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'create_server_multiple_hbas',
|
'create_server',
|
||||||
return_value=SCSERVER)
|
return_value=SCSERVER)
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'find_volume',
|
'find_volume',
|
||||||
@ -225,6 +225,69 @@ class DellSCSanFCDriverTestCase(test.TestCase):
|
|||||||
mock_find_volume.assert_called_once_with(fake.VOLUME_ID, None)
|
mock_find_volume.assert_called_once_with(fake.VOLUME_ID, None)
|
||||||
mock_get_volume.assert_called_once_with(self.VOLUME[u'instanceId'])
|
mock_get_volume.assert_called_once_with(self.VOLUME[u'instanceId'])
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_server',
|
||||||
|
return_value=SCSERVER)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_volume',
|
||||||
|
return_value=VOLUME)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'get_volume',
|
||||||
|
return_value=VOLUME)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'map_volume',
|
||||||
|
return_value=MAPPING)
|
||||||
|
@mock.patch.object(dell_storagecenter_fc.DellStorageCenterFCDriver,
|
||||||
|
'_is_live_vol')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_wwns')
|
||||||
|
@mock.patch.object(dell_storagecenter_fc.DellStorageCenterFCDriver,
|
||||||
|
'initialize_secondary')
|
||||||
|
def test_initialize_connection_live_vol(self,
|
||||||
|
mock_initialize_secondary,
|
||||||
|
mock_find_wwns,
|
||||||
|
mock_is_live_volume,
|
||||||
|
mock_map_volume,
|
||||||
|
mock_get_volume,
|
||||||
|
mock_find_volume,
|
||||||
|
mock_find_server,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID}
|
||||||
|
connector = self.connector
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_is_live_volume.return_value = sclivevol
|
||||||
|
mock_find_wwns.return_value = (
|
||||||
|
1, [u'5000D31000FCBE3D', u'5000D31000FCBE35'],
|
||||||
|
{u'21000024FF30441C': [u'5000D31000FCBE35'],
|
||||||
|
u'21000024FF30441D': [u'5000D31000FCBE3D']})
|
||||||
|
mock_initialize_secondary.return_value = (
|
||||||
|
1, [u'5000D31000FCBE3E', u'5000D31000FCBE36'],
|
||||||
|
{u'21000024FF30441E': [u'5000D31000FCBE36'],
|
||||||
|
u'21000024FF30441F': [u'5000D31000FCBE3E']})
|
||||||
|
res = self.driver.initialize_connection(volume, connector)
|
||||||
|
expected = {'data':
|
||||||
|
{'discard': True,
|
||||||
|
'initiator_target_map':
|
||||||
|
{u'21000024FF30441C': [u'5000D31000FCBE35'],
|
||||||
|
u'21000024FF30441D': [u'5000D31000FCBE3D'],
|
||||||
|
u'21000024FF30441E': [u'5000D31000FCBE36'],
|
||||||
|
u'21000024FF30441F': [u'5000D31000FCBE3E']},
|
||||||
|
'target_discovered': True,
|
||||||
|
'target_lun': 1,
|
||||||
|
'target_wwn': [u'5000D31000FCBE3D', u'5000D31000FCBE35',
|
||||||
|
u'5000D31000FCBE3E', u'5000D31000FCBE36']},
|
||||||
|
'driver_volume_type': 'fibre_channel'}
|
||||||
|
|
||||||
|
self.assertEqual(expected, res, 'Unexpected return data')
|
||||||
|
# verify find_volume has been called and that is has been called twice
|
||||||
|
mock_find_volume.assert_called_once_with(fake.VOLUME_ID, None)
|
||||||
|
mock_get_volume.assert_called_once_with(self.VOLUME[u'instanceId'])
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'find_server',
|
'find_server',
|
||||||
return_value=SCSERVER)
|
return_value=SCSERVER)
|
||||||
@ -260,7 +323,7 @@ class DellSCSanFCDriverTestCase(test.TestCase):
|
|||||||
'find_server',
|
'find_server',
|
||||||
return_value=None)
|
return_value=None)
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'create_server_multiple_hbas',
|
'create_server',
|
||||||
return_value=None)
|
return_value=None)
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'find_volume',
|
'find_volume',
|
||||||
@ -342,6 +405,101 @@ class DellSCSanFCDriverTestCase(test.TestCase):
|
|||||||
volume,
|
volume,
|
||||||
connector)
|
connector)
|
||||||
|
|
||||||
|
def test_initialize_secondary(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(
|
||||||
|
return_value=self.VOLUME)
|
||||||
|
find_wwns_ret = (1, [u'5000D31000FCBE3D', u'5000D31000FCBE35'],
|
||||||
|
{u'21000024FF30441C': [u'5000D31000FCBE35'],
|
||||||
|
u'21000024FF30441D': [u'5000D31000FCBE3D']})
|
||||||
|
mock_api.find_wwns = mock.MagicMock(return_value=find_wwns_ret)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=self.VOLUME)
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol,
|
||||||
|
['wwn1', 'wwn2'])
|
||||||
|
|
||||||
|
self.assertEqual(find_wwns_ret, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_create_server(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=None)
|
||||||
|
mock_api.create_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(
|
||||||
|
return_value=self.VOLUME)
|
||||||
|
find_wwns_ret = (1, [u'5000D31000FCBE3D', u'5000D31000FCBE35'],
|
||||||
|
{u'21000024FF30441C': [u'5000D31000FCBE35'],
|
||||||
|
u'21000024FF30441D': [u'5000D31000FCBE3D']})
|
||||||
|
mock_api.find_wwns = mock.MagicMock(return_value=find_wwns_ret)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=self.VOLUME)
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol,
|
||||||
|
['wwn1', 'wwn2'])
|
||||||
|
self.assertEqual(find_wwns_ret, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_no_server(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=None)
|
||||||
|
mock_api.create_server = mock.MagicMock(return_value=None)
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol,
|
||||||
|
['wwn1', 'wwn2'])
|
||||||
|
expected = (None, [], {})
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_map_fail(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(return_value=None)
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol,
|
||||||
|
['wwn1', 'wwn2'])
|
||||||
|
expected = (None, [], {})
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_vol_not_found(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(
|
||||||
|
return_value=self.VOLUME)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=None)
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol,
|
||||||
|
['wwn1', 'wwn2'])
|
||||||
|
expected = (None, [], {})
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'find_server',
|
'find_server',
|
||||||
return_value=SCSERVER)
|
return_value=SCSERVER)
|
||||||
@ -380,6 +538,56 @@ class DellSCSanFCDriverTestCase(test.TestCase):
|
|||||||
'data': {}}
|
'data': {}}
|
||||||
self.assertEqual(expected, res, 'Unexpected return data')
|
self.assertEqual(expected, res, 'Unexpected return data')
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_server',
|
||||||
|
return_value=SCSERVER)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_volume',
|
||||||
|
return_value=VOLUME)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'unmap_volume',
|
||||||
|
return_value=True)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_wwns',
|
||||||
|
return_value=(1,
|
||||||
|
[u'5000D31000FCBE3D',
|
||||||
|
u'5000D31000FCBE35'],
|
||||||
|
{u'21000024FF30441C':
|
||||||
|
[u'5000D31000FCBE35'],
|
||||||
|
u'21000024FF30441D':
|
||||||
|
[u'5000D31000FCBE3D']}))
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'get_volume_count',
|
||||||
|
return_value=1)
|
||||||
|
@mock.patch.object(dell_storagecenter_fc.DellStorageCenterFCDriver,
|
||||||
|
'_is_live_vol')
|
||||||
|
@mock.patch.object(dell_storagecenter_fc.DellStorageCenterFCDriver,
|
||||||
|
'terminate_secondary')
|
||||||
|
def test_terminate_connection_live_vol(self,
|
||||||
|
mock_terminate_secondary,
|
||||||
|
mock_is_live_vol,
|
||||||
|
mock_get_volume_count,
|
||||||
|
mock_find_wwns,
|
||||||
|
mock_unmap_volume,
|
||||||
|
mock_find_volume,
|
||||||
|
mock_find_server,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID}
|
||||||
|
connector = self.connector
|
||||||
|
mock_terminate_secondary.return_value = (None, [], {})
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_is_live_vol.return_value = sclivevol
|
||||||
|
res = self.driver.terminate_connection(volume, connector)
|
||||||
|
mock_unmap_volume.assert_called_once_with(self.VOLUME, self.SCSERVER)
|
||||||
|
expected = {'driver_volume_type': 'fibre_channel',
|
||||||
|
'data': {}}
|
||||||
|
self.assertEqual(expected, res, 'Unexpected return data')
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'find_server',
|
'find_server',
|
||||||
return_value=None)
|
return_value=None)
|
||||||
@ -482,10 +690,6 @@ class DellSCSanFCDriverTestCase(test.TestCase):
|
|||||||
mock_init):
|
mock_init):
|
||||||
volume = {'id': fake.VOLUME_ID}
|
volume = {'id': fake.VOLUME_ID}
|
||||||
connector = self.connector
|
connector = self.connector
|
||||||
# self.assertRaises(exception.VolumeBackendAPIException,
|
|
||||||
# self.driver.terminate_connection,
|
|
||||||
# volume,
|
|
||||||
# connector)
|
|
||||||
res = self.driver.terminate_connection(volume, connector)
|
res = self.driver.terminate_connection(volume, connector)
|
||||||
expected = {'driver_volume_type': 'fibre_channel',
|
expected = {'driver_volume_type': 'fibre_channel',
|
||||||
'data': {}}
|
'data': {}}
|
||||||
@ -572,6 +776,24 @@ class DellSCSanFCDriverTestCase(test.TestCase):
|
|||||||
'driver_volume_type': 'fibre_channel'}
|
'driver_volume_type': 'fibre_channel'}
|
||||||
self.assertEqual(expected, res, 'Unexpected return data')
|
self.assertEqual(expected, res, 'Unexpected return data')
|
||||||
|
|
||||||
|
def test_terminate_secondary(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=self.VOLUME)
|
||||||
|
mock_api.find_wwns = mock.MagicMock(return_value=(None, [], {}))
|
||||||
|
mock_api.unmap_volume = mock.MagicMock(return_value=True)
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
ret = self.driver.terminate_secondary(mock_api, sclivevol,
|
||||||
|
['wwn1', 'wwn2'])
|
||||||
|
expected = (None, [], {})
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'get_storage_usage',
|
'get_storage_usage',
|
||||||
return_value={'availableSpace': 100, 'freeSpace': 50})
|
return_value={'availableSpace': 100, 'freeSpace': 50})
|
||||||
|
@ -193,6 +193,7 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
IQN = 'iqn.2002-03.com.compellent:5000D31000000001'
|
IQN = 'iqn.2002-03.com.compellent:5000D31000000001'
|
||||||
|
|
||||||
ISCSI_PROPERTIES = {'access_mode': 'rw',
|
ISCSI_PROPERTIES = {'access_mode': 'rw',
|
||||||
|
'discard': True,
|
||||||
'target_discovered': False,
|
'target_discovered': False,
|
||||||
'target_iqn':
|
'target_iqn':
|
||||||
u'iqn.2002-03.com.compellent:5000d31000fcbe43',
|
u'iqn.2002-03.com.compellent:5000d31000fcbe43',
|
||||||
@ -274,6 +275,54 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
self.fake_iqn)
|
self.fake_iqn)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_sc')
|
||||||
|
def test_check_for_setup_error(self,
|
||||||
|
mock_find_sc,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
# Fail, Fail due to repl partner not found, success.
|
||||||
|
mock_find_sc.side_effect = [exception.VolumeBackendAPIException(''),
|
||||||
|
10000,
|
||||||
|
12345,
|
||||||
|
exception.VolumeBackendAPIException(''),
|
||||||
|
10000,
|
||||||
|
12345,
|
||||||
|
67890]
|
||||||
|
|
||||||
|
# Find SC throws
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.driver.check_for_setup_error)
|
||||||
|
# Replication enabled but one backend is down.
|
||||||
|
self.driver.replication_enabled = True
|
||||||
|
self.driver.backends = [{'target_device_id': '12345',
|
||||||
|
'managed_backend_name': 'host@dell1',
|
||||||
|
'qosnode': 'cinderqos'},
|
||||||
|
{'target_device_id': '67890',
|
||||||
|
'managed_backend_name': 'host@dell2',
|
||||||
|
'qosnode': 'otherqos'}]
|
||||||
|
self.assertRaises(exception.InvalidHost,
|
||||||
|
self.driver.check_for_setup_error)
|
||||||
|
# Good run. Should run without exceptions.
|
||||||
|
self.driver.check_for_setup_error()
|
||||||
|
# failed over run
|
||||||
|
mock_find_sc.side_effect = None
|
||||||
|
mock_find_sc.reset_mock()
|
||||||
|
mock_find_sc.return_value = 10000
|
||||||
|
self.driver.failed_over = True
|
||||||
|
self.driver.check_for_setup_error()
|
||||||
|
# find sc should be called exactly once
|
||||||
|
mock_find_sc.assert_called_once_with()
|
||||||
|
# No repl run
|
||||||
|
mock_find_sc.reset_mock()
|
||||||
|
mock_find_sc.return_value = 10000
|
||||||
|
self.driver.failed_over = False
|
||||||
|
self.driver.replication_enabled = False
|
||||||
|
self.driver.backends = None
|
||||||
|
self.driver.check_for_setup_error()
|
||||||
|
mock_find_sc.assert_called_once_with()
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
'_get_volume_extra_specs')
|
'_get_volume_extra_specs')
|
||||||
def test__create_replications(self,
|
def test__create_replications(self,
|
||||||
@ -358,6 +407,71 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
self.assertEqual({}, res)
|
self.assertEqual({}, res)
|
||||||
self.driver.backends = backends
|
self.driver.backends = backends
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_volume_extra_specs')
|
||||||
|
def test__create_replications_live_volume(self,
|
||||||
|
mock_get_volume_extra_specs,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
backends = self.driver.backends
|
||||||
|
model_update = {'replication_status': 'enabled',
|
||||||
|
'replication_driver_data': '12345'}
|
||||||
|
|
||||||
|
vol = {'id': fake.VOLUME_ID, 'replication_driver_data': ''}
|
||||||
|
scvol = {'name': fake.VOLUME_ID}
|
||||||
|
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.create_live_volume = mock.MagicMock(
|
||||||
|
return_value={'instanceId': '1'})
|
||||||
|
# Live volume with two backends defined.
|
||||||
|
self.driver.backends = [{'target_device_id': '12345',
|
||||||
|
'managed_backend_name': 'host@dell1',
|
||||||
|
'qosnode': 'cinderqos',
|
||||||
|
'remoteqos': 'remoteqos'},
|
||||||
|
{'target_device_id': '67890',
|
||||||
|
'managed_backend_name': 'host@dell2',
|
||||||
|
'qosnode': 'otherqos',
|
||||||
|
'remoteqos': 'remoteqos'}]
|
||||||
|
mock_get_volume_extra_specs.return_value = {
|
||||||
|
'replication:activereplay': '<is> True',
|
||||||
|
'replication_enabled': '<is> True',
|
||||||
|
'replication:livevolume': '<is> True'}
|
||||||
|
self.assertRaises(exception.ReplicationError,
|
||||||
|
self.driver._create_replications,
|
||||||
|
mock_api,
|
||||||
|
vol,
|
||||||
|
scvol)
|
||||||
|
# Live volume
|
||||||
|
self.driver.backends = [{'target_device_id': '12345',
|
||||||
|
'managed_backend_name': 'host@dell1',
|
||||||
|
'qosnode': 'cinderqos',
|
||||||
|
'diskfolder': 'ssd',
|
||||||
|
'remoteqos': 'remoteqos'}]
|
||||||
|
res = self.driver._create_replications(mock_api, vol, scvol)
|
||||||
|
mock_api.create_live_volume.assert_called_once_with(
|
||||||
|
scvol, '12345', True, False, 'cinderqos', 'remoteqos')
|
||||||
|
self.assertEqual(model_update, res)
|
||||||
|
# Active replay False
|
||||||
|
mock_get_volume_extra_specs.return_value = {
|
||||||
|
'replication_enabled': '<is> True',
|
||||||
|
'replication:livevolume': '<is> True'}
|
||||||
|
res = self.driver._create_replications(mock_api, vol, scvol)
|
||||||
|
mock_api.create_live_volume.assert_called_with(
|
||||||
|
scvol, '12345', False, False, 'cinderqos', 'remoteqos')
|
||||||
|
self.assertEqual(model_update, res)
|
||||||
|
# Sync
|
||||||
|
mock_get_volume_extra_specs.return_value = {
|
||||||
|
'replication_enabled': '<is> True',
|
||||||
|
'replication:livevolume': '<is> True',
|
||||||
|
'replication_type': '<in> sync'}
|
||||||
|
res = self.driver._create_replications(mock_api, vol, scvol)
|
||||||
|
mock_api.create_live_volume.assert_called_with(
|
||||||
|
scvol, '12345', False, True, 'cinderqos', 'remoteqos')
|
||||||
|
self.assertEqual(model_update, res)
|
||||||
|
|
||||||
|
self.driver.backends = backends
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
'_get_volume_extra_specs')
|
'_get_volume_extra_specs')
|
||||||
def test__delete_replications(self,
|
def test__delete_replications(self,
|
||||||
@ -388,6 +502,44 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
mock_api.delete_replication.assert_any_call(scvol, 67890)
|
mock_api.delete_replication.assert_any_call(scvol, 67890)
|
||||||
self.driver.backends = backends
|
self.driver.backends = backends
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_volume_extra_specs')
|
||||||
|
def test__delete_live_volume(self,
|
||||||
|
mock_get_volume_extra_specs,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
backends = self.driver.backends
|
||||||
|
vol = {'id': fake.VOLUME_ID}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api.get_live_volume = mock.MagicMock(return_value=sclivevol)
|
||||||
|
# No replication driver data.
|
||||||
|
ret = self.driver._delete_live_volume(mock_api, vol)
|
||||||
|
self.assertFalse(ret)
|
||||||
|
# Bogus rdd
|
||||||
|
vol = {'id': fake.VOLUME_ID, 'replication_driver_data': ''}
|
||||||
|
ret = self.driver._delete_live_volume(mock_api, vol)
|
||||||
|
self.assertFalse(ret)
|
||||||
|
# Valid delete.
|
||||||
|
mock_api.delete_live_volume = mock.MagicMock(return_value=True)
|
||||||
|
vol = {'id': fake.VOLUME_ID, 'replication_driver_data': '102'}
|
||||||
|
ret = self.driver._delete_live_volume(mock_api, vol)
|
||||||
|
self.assertTrue(ret)
|
||||||
|
# Wrong ssn.
|
||||||
|
vol = {'id': fake.VOLUME_ID, 'replication_driver_data': '103'}
|
||||||
|
ret = self.driver._delete_live_volume(mock_api, vol)
|
||||||
|
self.assertFalse(ret)
|
||||||
|
# No live volume found.
|
||||||
|
mock_api.get_live_volume.return_value = None
|
||||||
|
ret = self.driver._delete_live_volume(mock_api, vol)
|
||||||
|
self.assertFalse(ret)
|
||||||
|
|
||||||
|
self.driver.backends = backends
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'create_volume',
|
'create_volume',
|
||||||
return_value=VOLUME)
|
return_value=VOLUME)
|
||||||
@ -530,7 +682,12 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'delete_volume',
|
'delete_volume',
|
||||||
return_value=True)
|
return_value=True)
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs',
|
||||||
|
return_value={'enabled': True,
|
||||||
|
'live': False})
|
||||||
def test_delete_volume(self,
|
def test_delete_volume(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
mock_delete_volume,
|
mock_delete_volume,
|
||||||
mock_delete_replications,
|
mock_delete_replications,
|
||||||
mock_close_connection,
|
mock_close_connection,
|
||||||
@ -547,12 +704,38 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
self.assertTrue(mock_delete_replications.called)
|
self.assertTrue(mock_delete_replications.called)
|
||||||
self.assertEqual(2, mock_delete_replications.call_count)
|
self.assertEqual(2, mock_delete_replications.call_count)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_delete_live_volume')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'delete_volume',
|
||||||
|
return_value=True)
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs',
|
||||||
|
return_value={'enabled': True,
|
||||||
|
'live': True})
|
||||||
|
def test_delete_volume_live_volume(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
|
mock_delete_volume,
|
||||||
|
mock_delete_live_volume,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID, 'provider_id': '1.1'}
|
||||||
|
self.driver.delete_volume(volume)
|
||||||
|
mock_delete_volume.assert_called_with(fake.VOLUME_ID, '1.1')
|
||||||
|
self.assertTrue(mock_delete_live_volume.called)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
'_delete_replications')
|
'_delete_replications')
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'delete_volume',
|
'delete_volume',
|
||||||
return_value=False)
|
return_value=False)
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs',
|
||||||
|
return_value={'enabled': True,
|
||||||
|
'live': False})
|
||||||
def test_delete_volume_failure(self,
|
def test_delete_volume_failure(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
mock_delete_volume,
|
mock_delete_volume,
|
||||||
mock_delete_replications,
|
mock_delete_replications,
|
||||||
mock_close_connection,
|
mock_close_connection,
|
||||||
@ -644,7 +827,7 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
# verify find_volume has been called and that is has been called twice
|
# verify find_volume has been called and that is has been called twice
|
||||||
mock_find_volume.called_once_with(fake.VOLUME_ID, provider_id)
|
mock_find_volume.called_once_with(fake.VOLUME_ID, provider_id)
|
||||||
mock_get_volume.called_once_with(provider_id)
|
mock_get_volume.called_once_with(provider_id)
|
||||||
props = self.ISCSI_PROPERTIES
|
props = self.ISCSI_PROPERTIES.copy()
|
||||||
expected = {'data': props,
|
expected = {'data': props,
|
||||||
'driver_volume_type': 'iscsi'}
|
'driver_volume_type': 'iscsi'}
|
||||||
self.assertEqual(expected, data, 'Unexpected return value')
|
self.assertEqual(expected, data, 'Unexpected return value')
|
||||||
@ -767,6 +950,236 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
volume,
|
volume,
|
||||||
connector)
|
connector)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_server',
|
||||||
|
return_value=None)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'create_server',
|
||||||
|
return_value=SCSERVER)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_volume',
|
||||||
|
return_value=VOLUME)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'get_volume',
|
||||||
|
return_value=VOLUME)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'map_volume',
|
||||||
|
return_value=MAPPINGS[0])
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_iscsi_properties',
|
||||||
|
return_value=ISCSI_PROPERTIES)
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_is_live_vol')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'initialize_secondary')
|
||||||
|
def test_initialize_connection_live_volume(self,
|
||||||
|
mock_initialize_secondary,
|
||||||
|
mock_is_live_vol,
|
||||||
|
mock_find_iscsi_props,
|
||||||
|
mock_map_volume,
|
||||||
|
mock_get_volume,
|
||||||
|
mock_find_volume,
|
||||||
|
mock_create_server,
|
||||||
|
mock_find_server,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID}
|
||||||
|
connector = self.connector
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_is_live_vol.return_value = sclivevol
|
||||||
|
lvol_properties = {'access_mode': 'rw',
|
||||||
|
'target_discovered': False,
|
||||||
|
'target_iqn':
|
||||||
|
u'iqn:1',
|
||||||
|
'target_iqns':
|
||||||
|
[
|
||||||
|
u'iqn:1',
|
||||||
|
u'iqn:2'],
|
||||||
|
'target_lun': 1,
|
||||||
|
'target_luns': [1, 1],
|
||||||
|
'target_portal': u'192.168.1.21:3260',
|
||||||
|
'target_portals': [u'192.168.1.21:3260',
|
||||||
|
u'192.168.1.22:3260']}
|
||||||
|
mock_initialize_secondary.return_value = lvol_properties
|
||||||
|
props = self.ISCSI_PROPERTIES.copy()
|
||||||
|
props['target_iqns'] += lvol_properties['target_iqns']
|
||||||
|
props['target_luns'] += lvol_properties['target_luns']
|
||||||
|
props['target_portals'] += lvol_properties['target_portals']
|
||||||
|
ret = self.driver.initialize_connection(volume, connector)
|
||||||
|
expected = {'data': props,
|
||||||
|
'driver_volume_type': 'iscsi'}
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs',
|
||||||
|
return_value={'enabled': True, 'live': True})
|
||||||
|
def test_is_live_vol(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID,
|
||||||
|
'provider_id': '101.1'}
|
||||||
|
sclivevol = {'instanceId': '101.101'}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.get_live_volume = mock.MagicMock(return_value=sclivevol)
|
||||||
|
ret = self.driver._is_live_vol(mock_api, volume)
|
||||||
|
self.assertEqual(sclivevol, ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs',
|
||||||
|
return_value={'enabled': True, 'live': False})
|
||||||
|
def test_is_live_vol_repl_not_live(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
ret = self.driver._is_live_vol(mock_api, volume)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs',
|
||||||
|
return_value={'enabled': False, 'live': False})
|
||||||
|
def test_is_live_vol_no_repl(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
ret = self.driver._is_live_vol(mock_api, volume)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(
|
||||||
|
return_value=self.VOLUME)
|
||||||
|
mock_api.find_iscsi_properties = mock.MagicMock(
|
||||||
|
return_value=self.ISCSI_PROPERTIES)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=self.VOLUME)
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol, 'iqn')
|
||||||
|
self.assertEqual(self.ISCSI_PROPERTIES, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_create_server(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=None)
|
||||||
|
mock_api.create_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(
|
||||||
|
return_value=self.VOLUME)
|
||||||
|
mock_api.find_iscsi_properties = mock.MagicMock(
|
||||||
|
return_value=self.ISCSI_PROPERTIES)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=self.VOLUME)
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol, 'iqn')
|
||||||
|
self.assertEqual(self.ISCSI_PROPERTIES, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_no_server(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=None)
|
||||||
|
mock_api.create_server = mock.MagicMock(return_value=None)
|
||||||
|
expected = {'target_discovered': False,
|
||||||
|
'target_iqn': None,
|
||||||
|
'target_iqns': [],
|
||||||
|
'target_portal': None,
|
||||||
|
'target_portals': [],
|
||||||
|
'target_lun': None,
|
||||||
|
'target_luns': [],
|
||||||
|
}
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol, 'iqn')
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_map_fail(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(return_value=None)
|
||||||
|
expected = {'target_discovered': False,
|
||||||
|
'target_iqn': None,
|
||||||
|
'target_iqns': [],
|
||||||
|
'target_portal': None,
|
||||||
|
'target_portals': [],
|
||||||
|
'target_lun': None,
|
||||||
|
'target_luns': [],
|
||||||
|
}
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol, 'iqn')
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
|
def test_initialize_secondary_vol_not_found(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.map_secondary_volume = mock.MagicMock(
|
||||||
|
return_value=self.VOLUME)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=None)
|
||||||
|
expected = {'target_discovered': False,
|
||||||
|
'target_iqn': None,
|
||||||
|
'target_iqns': [],
|
||||||
|
'target_portal': None,
|
||||||
|
'target_portals': [],
|
||||||
|
'target_lun': None,
|
||||||
|
'target_luns': [],
|
||||||
|
}
|
||||||
|
ret = self.driver.initialize_secondary(mock_api, sclivevol, 'iqn')
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
|
def test_terminate_secondary(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.find_server = mock.MagicMock(return_value=self.SCSERVER)
|
||||||
|
mock_api.get_volume = mock.MagicMock(return_value=self.VOLUME)
|
||||||
|
mock_api.unmap_volume = mock.MagicMock()
|
||||||
|
self.driver.terminate_secondary(mock_api, sclivevol, 'iqn')
|
||||||
|
mock_api.find_server.assert_called_once_with('iqn', 102)
|
||||||
|
mock_api.get_volume.assert_called_once_with('102.101')
|
||||||
|
mock_api.unmap_volume.assert_called_once_with(self.VOLUME,
|
||||||
|
self.SCSERVER)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'find_server',
|
'find_server',
|
||||||
return_value=SCSERVER)
|
return_value=SCSERVER)
|
||||||
@ -789,6 +1202,40 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
mock_unmap_volume.assert_called_once_with(self.VOLUME, self.SCSERVER)
|
mock_unmap_volume.assert_called_once_with(self.VOLUME, self.SCSERVER)
|
||||||
self.assertIsNone(res, 'None expected')
|
self.assertIsNone(res, 'None expected')
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_server',
|
||||||
|
return_value=SCSERVER)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_volume',
|
||||||
|
return_value=VOLUME)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'unmap_volume',
|
||||||
|
return_value=True)
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_is_live_vol')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'terminate_secondary')
|
||||||
|
def test_terminate_connection_live_volume(self,
|
||||||
|
mock_terminate_secondary,
|
||||||
|
mock_is_live_vol,
|
||||||
|
mock_unmap_volume,
|
||||||
|
mock_find_volume,
|
||||||
|
mock_find_server,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
volume = {'id': fake.VOLUME_ID}
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_is_live_vol.return_value = sclivevol
|
||||||
|
connector = self.connector
|
||||||
|
res = self.driver.terminate_connection(volume, connector)
|
||||||
|
mock_unmap_volume.assert_called_once_with(self.VOLUME, self.SCSERVER)
|
||||||
|
self.assertIsNone(res, 'None expected')
|
||||||
|
self.assertTrue(mock_terminate_secondary.called)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'find_server',
|
'find_server',
|
||||||
return_value=None)
|
return_value=None)
|
||||||
@ -2201,8 +2648,57 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
self.assertIsNone(destssn)
|
self.assertIsNone(destssn)
|
||||||
self.driver.backends = backends
|
self.driver.backends = backends
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
def test__failover_live_volume(self,
|
||||||
'break_replication')
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
mock_api.get_live_volume = mock.MagicMock(return_value=sclivevol)
|
||||||
|
# Good run.
|
||||||
|
mock_api.swap_roles_live_volume = mock.MagicMock(return_value=True)
|
||||||
|
model_update = {'provider_id': '102.101',
|
||||||
|
'replication_status': 'failed-over'}
|
||||||
|
ret = self.driver._failover_live_volume(mock_api, fake.VOLUME_ID,
|
||||||
|
'101.100')
|
||||||
|
self.assertEqual(model_update, ret)
|
||||||
|
# Swap fail
|
||||||
|
mock_api.swap_roles_live_volume.return_value = False
|
||||||
|
model_update = {'status': 'error'}
|
||||||
|
ret = self.driver._failover_live_volume(mock_api, fake.VOLUME_ID,
|
||||||
|
'101.100')
|
||||||
|
self.assertEqual(model_update, ret)
|
||||||
|
# Can't find live volume.
|
||||||
|
mock_api.get_live_volume.return_value = None
|
||||||
|
ret = self.driver._failover_live_volume(mock_api, fake.VOLUME_ID,
|
||||||
|
'101.100')
|
||||||
|
self.assertEqual(model_update, ret)
|
||||||
|
|
||||||
|
def test__failover_replication(self,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
rvol = {'instanceId': '102.101'}
|
||||||
|
mock_api = mock.MagicMock()
|
||||||
|
mock_api.break_replication = mock.MagicMock(return_value=rvol)
|
||||||
|
# Good run.
|
||||||
|
model_update = {'replication_status': 'failed-over',
|
||||||
|
'provider_id': '102.101'}
|
||||||
|
ret = self.driver._failover_replication(mock_api, fake.VOLUME_ID,
|
||||||
|
'101.100', 102)
|
||||||
|
self.assertEqual(model_update, ret)
|
||||||
|
# break fail
|
||||||
|
mock_api.break_replication.return_value = None
|
||||||
|
model_update = {'status': 'error'}
|
||||||
|
ret = self.driver._failover_replication(mock_api, fake.VOLUME_ID,
|
||||||
|
'101.100', 102)
|
||||||
|
self.assertEqual(model_update, ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_failover_replication')
|
||||||
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
'_parse_secondary')
|
'_parse_secondary')
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
@ -2211,15 +2707,20 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
'remove_mappings')
|
'remove_mappings')
|
||||||
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
'failback_volumes')
|
'failback_volumes')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs')
|
||||||
def test_failover_host(self,
|
def test_failover_host(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
mock_failback_volumes,
|
mock_failback_volumes,
|
||||||
mock_remove_mappings,
|
mock_remove_mappings,
|
||||||
mock_find_volume,
|
mock_find_volume,
|
||||||
mock_parse_secondary,
|
mock_parse_secondary,
|
||||||
mock_break_replication,
|
mock_failover_replication,
|
||||||
mock_close_connection,
|
mock_close_connection,
|
||||||
mock_open_connection,
|
mock_open_connection,
|
||||||
mock_init):
|
mock_init):
|
||||||
|
mock_get_replication_specs.return_value = {'enabled': False,
|
||||||
|
'live': False}
|
||||||
self.driver.replication_enabled = False
|
self.driver.replication_enabled = False
|
||||||
self.driver.failed_over = False
|
self.driver.failed_over = False
|
||||||
volumes = [{'id': fake.VOLUME_ID,
|
volumes = [{'id': fake.VOLUME_ID,
|
||||||
@ -2236,12 +2737,133 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
'12345')
|
'12345')
|
||||||
# Good run
|
# Good run
|
||||||
self.driver.replication_enabled = True
|
self.driver.replication_enabled = True
|
||||||
|
mock_get_replication_specs.return_value = {'enabled': True,
|
||||||
|
'live': False}
|
||||||
mock_parse_secondary.return_value = 12345
|
mock_parse_secondary.return_value = 12345
|
||||||
expected_destssn = 12345
|
expected_destssn = 12345
|
||||||
mock_break_replication.side_effect = [{'instanceId': '2.1'}, # test1
|
mock_failover_replication.side_effect = [
|
||||||
{'instanceId': '2.2'},
|
{'provider_id': '2.1', 'replication_status': 'failed-over'}, # 1
|
||||||
{'instanceId': '2.1'}, # test2
|
{'provider_id': '2.2', 'replication_status': 'failed-over'},
|
||||||
{'instanceId': '2.1'}] # test3
|
{'provider_id': '2.1', 'replication_status': 'failed-over'}, # 2
|
||||||
|
{'provider_id': '2.1', 'replication_status': 'failed-over'}] # 3
|
||||||
|
expected_volume_update = [{'volume_id': fake.VOLUME_ID, 'updates':
|
||||||
|
{'replication_status': 'failed-over',
|
||||||
|
'provider_id': '2.1'}},
|
||||||
|
{'volume_id': fake.VOLUME2_ID, 'updates':
|
||||||
|
{'replication_status': 'failed-over',
|
||||||
|
'provider_id': '2.2'}}]
|
||||||
|
destssn, volume_update = self.driver.failover_host(
|
||||||
|
{}, volumes, '12345')
|
||||||
|
self.assertEqual(expected_destssn, destssn)
|
||||||
|
self.assertEqual(expected_volume_update, volume_update)
|
||||||
|
# Good run. Not all volumes replicated.
|
||||||
|
volumes = [{'id': fake.VOLUME_ID, 'replication_driver_data': '12345'},
|
||||||
|
{'id': fake.VOLUME2_ID, 'replication_driver_data': ''}]
|
||||||
|
expected_volume_update = [{'volume_id': fake.VOLUME_ID, 'updates':
|
||||||
|
{'replication_status': 'failed-over',
|
||||||
|
'provider_id': '2.1'}},
|
||||||
|
{'volume_id': fake.VOLUME2_ID, 'updates':
|
||||||
|
{'status': 'error'}}]
|
||||||
|
self.driver.failed_over = False
|
||||||
|
self.driver.active_backend_id = None
|
||||||
|
destssn, volume_update = self.driver.failover_host(
|
||||||
|
{}, volumes, '12345')
|
||||||
|
self.assertEqual(expected_destssn, destssn)
|
||||||
|
self.assertEqual(expected_volume_update, volume_update)
|
||||||
|
# Good run. Not all volumes replicated. No replication_driver_data.
|
||||||
|
volumes = [{'id': fake.VOLUME_ID, 'replication_driver_data': '12345'},
|
||||||
|
{'id': fake.VOLUME2_ID}]
|
||||||
|
expected_volume_update = [{'volume_id': fake.VOLUME_ID, 'updates':
|
||||||
|
{'replication_status': 'failed-over',
|
||||||
|
'provider_id': '2.1'}},
|
||||||
|
{'volume_id': fake.VOLUME2_ID, 'updates':
|
||||||
|
{'status': 'error'}}]
|
||||||
|
self.driver.failed_over = False
|
||||||
|
self.driver.active_backend_id = None
|
||||||
|
destssn, volume_update = self.driver.failover_host(
|
||||||
|
{}, volumes, '12345')
|
||||||
|
self.assertEqual(expected_destssn, destssn)
|
||||||
|
self.assertEqual(expected_volume_update, volume_update)
|
||||||
|
# Good run. No volumes replicated. No replication_driver_data.
|
||||||
|
volumes = [{'id': fake.VOLUME_ID},
|
||||||
|
{'id': fake.VOLUME2_ID}]
|
||||||
|
expected_volume_update = [{'volume_id': fake.VOLUME_ID, 'updates':
|
||||||
|
{'status': 'error'}},
|
||||||
|
{'volume_id': fake.VOLUME2_ID, 'updates':
|
||||||
|
{'status': 'error'}}]
|
||||||
|
self.driver.failed_over = False
|
||||||
|
self.driver.active_backend_id = None
|
||||||
|
destssn, volume_update = self.driver.failover_host(
|
||||||
|
{}, volumes, '12345')
|
||||||
|
self.assertEqual(expected_destssn, destssn)
|
||||||
|
self.assertEqual(expected_volume_update, volume_update)
|
||||||
|
# Secondary not found.
|
||||||
|
mock_parse_secondary.return_value = None
|
||||||
|
self.driver.failed_over = False
|
||||||
|
self.driver.active_backend_id = None
|
||||||
|
self.assertRaises(exception.InvalidInput,
|
||||||
|
self.driver.failover_host,
|
||||||
|
{},
|
||||||
|
volumes,
|
||||||
|
'54321')
|
||||||
|
# Already failed over.
|
||||||
|
self.driver.failed_over = True
|
||||||
|
self.driver.failover_host({}, volumes, 'default')
|
||||||
|
mock_failback_volumes.assert_called_once_with(volumes)
|
||||||
|
# Already failed over.
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.driver.failover_host, {}, volumes, '67890')
|
||||||
|
self.driver.replication_enabled = False
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_failover_live_volume')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_parse_secondary')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_volume')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'remove_mappings')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'failback_volumes')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs')
|
||||||
|
def test_failover_host_live_volume(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
|
mock_failback_volumes,
|
||||||
|
mock_remove_mappings,
|
||||||
|
mock_find_volume,
|
||||||
|
mock_parse_secondary,
|
||||||
|
mock_failover_live_volume,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
mock_get_replication_specs.return_value = {'enabled': False,
|
||||||
|
'live': False}
|
||||||
|
self.driver.replication_enabled = False
|
||||||
|
self.driver.failed_over = False
|
||||||
|
volumes = [{'id': fake.VOLUME_ID,
|
||||||
|
'replication_driver_data': '12345',
|
||||||
|
'provider_id': '1.1'},
|
||||||
|
{'id': fake.VOLUME2_ID,
|
||||||
|
'replication_driver_data': '12345',
|
||||||
|
'provider_id': '1.2'}]
|
||||||
|
# No run. Not doing repl. Should raise.
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.driver.failover_host,
|
||||||
|
{},
|
||||||
|
volumes,
|
||||||
|
'12345')
|
||||||
|
# Good run
|
||||||
|
self.driver.replication_enabled = True
|
||||||
|
mock_get_replication_specs.return_value = {'enabled': True,
|
||||||
|
'live': True}
|
||||||
|
mock_parse_secondary.return_value = 12345
|
||||||
|
expected_destssn = 12345
|
||||||
|
mock_failover_live_volume.side_effect = [
|
||||||
|
{'provider_id': '2.1', 'replication_status': 'failed-over'}, # 1
|
||||||
|
{'provider_id': '2.2', 'replication_status': 'failed-over'},
|
||||||
|
{'provider_id': '2.1', 'replication_status': 'failed-over'}, # 2
|
||||||
|
{'provider_id': '2.1', 'replication_status': 'failed-over'}] # 3
|
||||||
expected_volume_update = [{'volume_id': fake.VOLUME_ID, 'updates':
|
expected_volume_update = [{'volume_id': fake.VOLUME_ID, 'updates':
|
||||||
{'replication_status': 'failed-over',
|
{'replication_status': 'failed-over',
|
||||||
'provider_id': '2.1'}},
|
'provider_id': '2.1'}},
|
||||||
@ -2531,6 +3153,77 @@ class DellSCSanISCSIDriverTestCase(test.TestCase):
|
|||||||
self.driver.failed_over = False
|
self.driver.failed_over = False
|
||||||
self.driver.backends = backends
|
self.driver.backends = backends
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_qos',
|
||||||
|
return_value='cinderqos')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_volume')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_update_backend')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'get_live_volume')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'swap_roles_live_volume')
|
||||||
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
|
'_get_replication_specs')
|
||||||
|
def test_failback_volumes_live_vol(self,
|
||||||
|
mock_get_replication_specs,
|
||||||
|
mock_swap_roles_live_volume,
|
||||||
|
mock_get_live_volume,
|
||||||
|
mock_update_backend,
|
||||||
|
mock_find_volume,
|
||||||
|
mock_get_qos,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
self.driver.replication_enabled = True
|
||||||
|
self.driver.failed_over = True
|
||||||
|
self.driver.active_backend_id = 12345
|
||||||
|
self.driver.primaryssn = 11111
|
||||||
|
backends = self.driver.backends
|
||||||
|
self.driver.backends = [{'target_device_id': '12345',
|
||||||
|
'qosnode': 'cinderqos',
|
||||||
|
'remoteqos': 'remoteqos'}]
|
||||||
|
volumes = [{'id': fake.VOLUME_ID,
|
||||||
|
'replication_driver_data': '12345',
|
||||||
|
'provider_id': '12345.1'},
|
||||||
|
{'id': fake.VOLUME2_ID,
|
||||||
|
'replication_driver_data': '12345',
|
||||||
|
'provider_id': '12345.2'}]
|
||||||
|
mock_get_live_volume.side_effect = [
|
||||||
|
{'instanceId': '11111.101',
|
||||||
|
'secondaryVolume': {'instanceId': '11111.1001',
|
||||||
|
'instanceName': fake.VOLUME_ID},
|
||||||
|
'secondaryScSerialNumber': 11111},
|
||||||
|
{'instanceId': '11111.102',
|
||||||
|
'secondaryVolume': {'instanceId': '11111.1002',
|
||||||
|
'instanceName': fake.VOLUME2_ID},
|
||||||
|
'secondaryScSerialNumber': 11111}
|
||||||
|
]
|
||||||
|
mock_get_replication_specs.return_value = {'enabled': True,
|
||||||
|
'live': True}
|
||||||
|
mock_swap_roles_live_volume.side_effect = [True, True]
|
||||||
|
mock_find_volume.side_effect = [{'instanceId': '12345.1'},
|
||||||
|
{'instanceId': '12345.2'}]
|
||||||
|
|
||||||
|
# we don't care about the return. We just want to make sure that
|
||||||
|
# _wait_for_replication is called with the proper replitems.
|
||||||
|
ret = self.driver.failback_volumes(volumes)
|
||||||
|
expected = [{'updates': {'provider_id': '11111.1001',
|
||||||
|
'replication_status': 'enabled',
|
||||||
|
'status': 'available'},
|
||||||
|
'volume_id': fake.VOLUME_ID},
|
||||||
|
{'updates': {'provider_id': '11111.1002',
|
||||||
|
'replication_status': 'enabled',
|
||||||
|
'status': 'available'},
|
||||||
|
'volume_id': fake.VOLUME2_ID}]
|
||||||
|
|
||||||
|
self.assertEqual(expected, ret)
|
||||||
|
|
||||||
|
self.driver.replication_enabled = False
|
||||||
|
self.driver.failed_over = False
|
||||||
|
self.driver.backends = backends
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
@mock.patch.object(dell_storagecenter_iscsi.DellStorageCenterISCSIDriver,
|
||||||
'_get_qos',
|
'_get_qos',
|
||||||
return_value='cinderqos')
|
return_value='cinderqos')
|
||||||
|
@ -1916,7 +1916,7 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
False)
|
False)
|
||||||
mock_find_folder.assert_called_once_with(
|
mock_find_folder.assert_called_once_with(
|
||||||
'StorageCenter/ScVolumeFolder/GetList',
|
'StorageCenter/ScVolumeFolder/GetList',
|
||||||
self.configuration.dell_sc_volume_folder)
|
self.configuration.dell_sc_volume_folder, -1)
|
||||||
self.assertIsNone(res, 'Expected None')
|
self.assertIsNone(res, 'Expected None')
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
@ -1931,7 +1931,7 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
False)
|
False)
|
||||||
mock_find_folder.assert_called_once_with(
|
mock_find_folder.assert_called_once_with(
|
||||||
'StorageCenter/ScVolumeFolder/GetList',
|
'StorageCenter/ScVolumeFolder/GetList',
|
||||||
self.configuration.dell_sc_volume_folder)
|
self.configuration.dell_sc_volume_folder, -1)
|
||||||
self.assertEqual(self.FLDR, res, 'Unexpected Folder')
|
self.assertEqual(self.FLDR, res, 'Unexpected Folder')
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
@ -2003,7 +2003,7 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
True)
|
True)
|
||||||
mock_find_folder.assert_called_once_with(
|
mock_find_folder.assert_called_once_with(
|
||||||
'StorageCenter/ScVolumeFolder/GetList',
|
'StorageCenter/ScVolumeFolder/GetList',
|
||||||
self.configuration.dell_sc_volume_folder)
|
self.configuration.dell_sc_volume_folder, -1)
|
||||||
self.assertTrue(mock_create_folder_path.called)
|
self.assertTrue(mock_create_folder_path.called)
|
||||||
self.assertEqual(self.FLDR, res, 'Unexpected Folder')
|
self.assertEqual(self.FLDR, res, 'Unexpected Folder')
|
||||||
|
|
||||||
@ -2546,7 +2546,7 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
res = self.scapi._find_server_folder(False)
|
res = self.scapi._find_server_folder(False)
|
||||||
mock_find_folder.assert_called_once_with(
|
mock_find_folder.assert_called_once_with(
|
||||||
'StorageCenter/ScServerFolder/GetList',
|
'StorageCenter/ScServerFolder/GetList',
|
||||||
self.configuration.dell_sc_server_folder)
|
self.configuration.dell_sc_server_folder, 12345)
|
||||||
self.assertEqual(self.SVR_FLDR, res, 'Unexpected server folder')
|
self.assertEqual(self.SVR_FLDR, res, 'Unexpected server folder')
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
@ -2566,7 +2566,7 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
res = self.scapi._find_server_folder(True)
|
res = self.scapi._find_server_folder(True)
|
||||||
mock_find_folder.assert_called_once_with(
|
mock_find_folder.assert_called_once_with(
|
||||||
'StorageCenter/ScServerFolder/GetList',
|
'StorageCenter/ScServerFolder/GetList',
|
||||||
self.configuration.dell_sc_server_folder)
|
self.configuration.dell_sc_server_folder, 12345)
|
||||||
self.assertTrue(mock_create_folder_path.called)
|
self.assertTrue(mock_create_folder_path.called)
|
||||||
self.assertEqual(self.SVR_FLDR, res, 'Unexpected server folder')
|
self.assertEqual(self.SVR_FLDR, res, 'Unexpected server folder')
|
||||||
|
|
||||||
@ -2583,7 +2583,7 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
False)
|
False)
|
||||||
mock_find_folder.assert_called_once_with(
|
mock_find_folder.assert_called_once_with(
|
||||||
'StorageCenter/ScServerFolder/GetList',
|
'StorageCenter/ScServerFolder/GetList',
|
||||||
self.configuration.dell_sc_volume_folder)
|
self.configuration.dell_sc_volume_folder, 12345)
|
||||||
self.assertIsNone(res, 'Expected None')
|
self.assertIsNone(res, 'Expected None')
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
@ -2674,20 +2674,23 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
res = self.scapi._find_serveros('Red Hat Linux 6.x')
|
res = self.scapi._find_serveros('Red Hat Linux 6.x')
|
||||||
self.assertIsNone(res, 'None expected')
|
self.assertIsNone(res, 'None expected')
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_find_server_folder',
|
||||||
|
return_value=SVR_FLDR)
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'_add_hba',
|
'_add_hba',
|
||||||
return_value=FC_HBA)
|
return_value=FC_HBA)
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'create_server',
|
'_create_server',
|
||||||
return_value=SCSERVER)
|
return_value=SCSERVER)
|
||||||
def test_create_server_multiple_hbas(self,
|
def test_create_server_multiple_hbas(self,
|
||||||
mock_create_server,
|
mock_create_server,
|
||||||
mock_add_hba,
|
mock_add_hba,
|
||||||
|
mock_find_server_folder,
|
||||||
mock_close_connection,
|
mock_close_connection,
|
||||||
mock_open_connection,
|
mock_open_connection,
|
||||||
mock_init):
|
mock_init):
|
||||||
res = self.scapi.create_server_multiple_hbas(
|
res = self.scapi.create_server(self.WWNS)
|
||||||
self.WWNS)
|
|
||||||
self.assertTrue(mock_create_server.called)
|
self.assertTrue(mock_create_server.called)
|
||||||
self.assertTrue(mock_add_hba.called)
|
self.assertTrue(mock_add_hba.called)
|
||||||
self.assertEqual(self.SCSERVER, res, 'Unexpected ScServer')
|
self.assertEqual(self.SCSERVER, res, 'Unexpected ScServer')
|
||||||
@ -3417,93 +3420,6 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
'target_portals': [u'192.168.0.21:3260']}
|
'target_portals': [u'192.168.0.21:3260']}
|
||||||
self.assertEqual(expected, res, 'Wrong Target Info')
|
self.assertEqual(expected, res, 'Wrong Target Info')
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_active_controller',
|
|
||||||
return_value='64702.64702')
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_controller_port',
|
|
||||||
return_value=ISCSI_CTRLR_PORT)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_domains',
|
|
||||||
return_value=ISCSI_FLT_DOMAINS)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_mappings',
|
|
||||||
return_value=MAPPINGS)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_is_virtualport_mode',
|
|
||||||
return_value=True)
|
|
||||||
def test_find_iscsi_properties_by_address(self,
|
|
||||||
mock_is_virtualport_mode,
|
|
||||||
mock_find_mappings,
|
|
||||||
mock_find_domains,
|
|
||||||
mock_find_ctrl_port,
|
|
||||||
mock_find_active_controller,
|
|
||||||
mock_close_connection,
|
|
||||||
mock_open_connection,
|
|
||||||
mock_init):
|
|
||||||
# Test case to find iSCSI mappings by IP Address & port
|
|
||||||
res = self.scapi.find_iscsi_properties(
|
|
||||||
self.VOLUME, '192.168.0.21', 3260)
|
|
||||||
self.assertTrue(mock_is_virtualport_mode.called)
|
|
||||||
self.assertTrue(mock_find_mappings.called)
|
|
||||||
self.assertTrue(mock_find_domains.called)
|
|
||||||
self.assertTrue(mock_find_ctrl_port.called)
|
|
||||||
self.assertTrue(mock_find_active_controller.called)
|
|
||||||
expected = {'target_discovered': False,
|
|
||||||
'target_iqn':
|
|
||||||
u'iqn.2002-03.com.compellent:5000d31000fcbe43',
|
|
||||||
'target_iqns':
|
|
||||||
[u'iqn.2002-03.com.compellent:5000d31000fcbe43'],
|
|
||||||
'target_lun': 1,
|
|
||||||
'target_luns': [1],
|
|
||||||
'target_portal': u'192.168.0.21:3260',
|
|
||||||
'target_portals': [u'192.168.0.21:3260']}
|
|
||||||
self.assertEqual(expected, res, 'Wrong Target Info')
|
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_active_controller',
|
|
||||||
return_value='64702.64702')
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_controller_port',
|
|
||||||
return_value=ISCSI_CTRLR_PORT)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_domains',
|
|
||||||
return_value=ISCSI_FLT_DOMAINS)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_mappings',
|
|
||||||
return_value=MAPPINGS)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_is_virtualport_mode',
|
|
||||||
return_value=True)
|
|
||||||
def test_find_iscsi_properties_by_address_not_found(
|
|
||||||
self,
|
|
||||||
mock_is_virtualport_mode,
|
|
||||||
mock_find_mappings,
|
|
||||||
mock_find_domains,
|
|
||||||
mock_find_ctrl_port,
|
|
||||||
mock_find_active_ctrl,
|
|
||||||
mock_close_connection,
|
|
||||||
mock_open_connection,
|
|
||||||
mock_init):
|
|
||||||
# Test case to find iSCSI mappings by IP Address & port are not found
|
|
||||||
res = self.scapi.find_iscsi_properties(
|
|
||||||
self.VOLUME, '192.168.1.21', 3260)
|
|
||||||
self.assertTrue(mock_is_virtualport_mode.called)
|
|
||||||
self.assertTrue(mock_find_mappings.called)
|
|
||||||
self.assertTrue(mock_find_domains.called)
|
|
||||||
self.assertTrue(mock_find_ctrl_port.called)
|
|
||||||
self.assertTrue(mock_find_active_ctrl.called)
|
|
||||||
expected = {'target_discovered': False,
|
|
||||||
'target_iqn':
|
|
||||||
u'iqn.2002-03.com.compellent:5000d31000fcbe43',
|
|
||||||
'target_iqns':
|
|
||||||
[u'iqn.2002-03.com.compellent:5000d31000fcbe43'],
|
|
||||||
'target_lun': 1,
|
|
||||||
'target_luns': [1],
|
|
||||||
'target_portal': u'192.168.0.21:3260',
|
|
||||||
'target_portals': [u'192.168.0.21:3260']}
|
|
||||||
self.assertEqual(expected, res, 'Wrong Target Info')
|
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'_find_mappings',
|
'_find_mappings',
|
||||||
return_value=[])
|
return_value=[])
|
||||||
@ -3747,94 +3663,6 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
self.assertTrue(mock_find_controller_port_iscsi_config.called)
|
self.assertTrue(mock_find_controller_port_iscsi_config.called)
|
||||||
self.assertTrue(mock_find_active_controller.called)
|
self.assertTrue(mock_find_active_controller.called)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_active_controller',
|
|
||||||
return_value='64702.64702')
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_controller_port',
|
|
||||||
return_value=ISCSI_CTRLR_PORT)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_mappings',
|
|
||||||
return_value=MAPPINGS)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_is_virtualport_mode',
|
|
||||||
return_value=False)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_controller_port_iscsi_config',
|
|
||||||
return_value=ISCSI_CONFIG)
|
|
||||||
def test_find_iscsi_properties_by_address_legacy(
|
|
||||||
self,
|
|
||||||
mock_find_controller_port_iscsi_config,
|
|
||||||
mock_is_virtualport_mode,
|
|
||||||
mock_find_mappings,
|
|
||||||
mock_find_ctrl_port,
|
|
||||||
mock_find_active_controller,
|
|
||||||
mock_close_connection,
|
|
||||||
mock_open_connection,
|
|
||||||
mock_init):
|
|
||||||
# Test case to find iSCSI mappings by IP Address & port
|
|
||||||
res = self.scapi.find_iscsi_properties(
|
|
||||||
self.VOLUME, '192.168.0.21', 3260)
|
|
||||||
self.assertTrue(mock_is_virtualport_mode.called)
|
|
||||||
self.assertTrue(mock_find_mappings.called)
|
|
||||||
self.assertTrue(mock_find_ctrl_port.called)
|
|
||||||
self.assertTrue(mock_find_active_controller.called)
|
|
||||||
self.assertTrue(mock_find_controller_port_iscsi_config.called)
|
|
||||||
expected = {'target_discovered': False,
|
|
||||||
'target_iqn':
|
|
||||||
u'iqn.2002-03.com.compellent:5000d31000fcbe43',
|
|
||||||
'target_iqns':
|
|
||||||
[u'iqn.2002-03.com.compellent:5000d31000fcbe43'],
|
|
||||||
'target_lun': 1,
|
|
||||||
'target_luns': [1],
|
|
||||||
'target_portal': u'192.168.0.21:3260',
|
|
||||||
'target_portals': [u'192.168.0.21:3260']}
|
|
||||||
self.assertEqual(expected, res, 'Wrong Target Info')
|
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_active_controller',
|
|
||||||
return_value='64702.64702')
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_controller_port',
|
|
||||||
return_value=ISCSI_CTRLR_PORT)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_mappings',
|
|
||||||
return_value=MAPPINGS)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_is_virtualport_mode',
|
|
||||||
return_value=False)
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
|
||||||
'_find_controller_port_iscsi_config',
|
|
||||||
return_value=ISCSI_CONFIG)
|
|
||||||
def test_find_iscsi_properties_by_address_not_found_legacy(
|
|
||||||
self,
|
|
||||||
mock_find_controller_port_iscsi_config,
|
|
||||||
mock_is_virtualport_mode,
|
|
||||||
mock_find_mappings,
|
|
||||||
mock_find_ctrl_port,
|
|
||||||
mock_find_active_ctrl,
|
|
||||||
mock_close_connection,
|
|
||||||
mock_open_connection,
|
|
||||||
mock_init):
|
|
||||||
# Test case to find iSCSI mappings by IP Address & port are not found
|
|
||||||
res = self.scapi.find_iscsi_properties(
|
|
||||||
self.VOLUME, '192.168.1.21', 3260)
|
|
||||||
self.assertTrue(mock_is_virtualport_mode.called)
|
|
||||||
self.assertTrue(mock_find_mappings.called)
|
|
||||||
self.assertTrue(mock_find_ctrl_port.called)
|
|
||||||
self.assertTrue(mock_find_active_ctrl.called)
|
|
||||||
self.assertTrue(mock_find_controller_port_iscsi_config.called)
|
|
||||||
expected = {'target_discovered': False,
|
|
||||||
'target_iqn':
|
|
||||||
u'iqn.2002-03.com.compellent:5000d31000fcbe43',
|
|
||||||
'target_iqns':
|
|
||||||
[u'iqn.2002-03.com.compellent:5000d31000fcbe43'],
|
|
||||||
'target_lun': 1,
|
|
||||||
'target_luns': [1],
|
|
||||||
'target_portal': u'192.168.0.21:3260',
|
|
||||||
'target_portals': [u'192.168.0.21:3260']}
|
|
||||||
self.assertEqual(expected, res, 'Wrong Target Info')
|
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
'_find_active_controller',
|
'_find_active_controller',
|
||||||
return_value='64702.64702')
|
return_value='64702.64702')
|
||||||
@ -5965,6 +5793,133 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
self.scapi._find_qos,
|
self.scapi._find_qos,
|
||||||
'Cinder QoS')
|
'Cinder QoS')
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'put',
|
||||||
|
return_value=RESPONSE_400)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'get',
|
||||||
|
return_value=RESPONSE_200)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json',
|
||||||
|
return_value=SCREPL)
|
||||||
|
def test_update_replicate_active_replay_fail(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_get,
|
||||||
|
mock_put,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
ret = self.scapi.update_replicate_active_replay({'instanceId': '1'},
|
||||||
|
True)
|
||||||
|
self.assertFalse(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'get',
|
||||||
|
return_value=RESPONSE_200)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json',
|
||||||
|
return_value=SCREPL)
|
||||||
|
def test_update_replicate_active_replay_nothing_to_do(
|
||||||
|
self, mock_get_json, mock_get, mock_close_connection,
|
||||||
|
mock_open_connection, mock_init):
|
||||||
|
ret = self.scapi.update_replicate_active_replay({'instanceId': '1'},
|
||||||
|
False)
|
||||||
|
self.assertTrue(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'get',
|
||||||
|
return_value=RESPONSE_200)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json',
|
||||||
|
return_value=[])
|
||||||
|
def test_update_replicate_active_replay_not_found(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_get,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
ret = self.scapi.update_replicate_active_replay({'instanceId': '1'},
|
||||||
|
True)
|
||||||
|
self.assertTrue(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'get',
|
||||||
|
return_value=RESPONSE_400)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json',
|
||||||
|
return_value=[])
|
||||||
|
def test_update_replicate_active_replay_not_found2(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_get,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
ret = self.scapi.update_replicate_active_replay({'instanceId': '1'},
|
||||||
|
True)
|
||||||
|
self.assertTrue(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'post',
|
||||||
|
return_value=RESPONSE_200)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json',
|
||||||
|
return_value=[{'instanceId': '12345.1'}])
|
||||||
|
def test_get_disk_folder(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_post,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
ret = self.scapi._get_disk_folder(12345, 'name')
|
||||||
|
expected_payload = {'filter': {'filterType': 'AND', 'filters': [
|
||||||
|
{'filterType': 'Equals', 'attributeName': 'scSerialNumber',
|
||||||
|
'attributeValue': 12345},
|
||||||
|
{'filterType': 'Equals', 'attributeName': 'name',
|
||||||
|
'attributeValue': 'name'}]}}
|
||||||
|
mock_post.assert_called_once_with('StorageCenter/ScDiskFolder/GetList',
|
||||||
|
expected_payload)
|
||||||
|
self.assertEqual({'instanceId': '12345.1'}, ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'post',
|
||||||
|
return_value=RESPONSE_400)
|
||||||
|
def test_get_disk_folder_fail(self,
|
||||||
|
mock_post,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
ret = self.scapi._get_disk_folder(12345, 'name')
|
||||||
|
expected_payload = {'filter': {'filterType': 'AND', 'filters': [
|
||||||
|
{'filterType': 'Equals', 'attributeName': 'scSerialNumber',
|
||||||
|
'attributeValue': 12345},
|
||||||
|
{'filterType': 'Equals', 'attributeName': 'name',
|
||||||
|
'attributeValue': 'name'}]}}
|
||||||
|
mock_post.assert_called_once_with('StorageCenter/ScDiskFolder/GetList',
|
||||||
|
expected_payload)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'post',
|
||||||
|
return_value=RESPONSE_200)
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json')
|
||||||
|
def test_get_disk_folder_fail_bad_json(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_post,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
mock_get_json.side_effect = (exception.VolumeBackendAPIException(''))
|
||||||
|
ret = self.scapi._get_disk_folder(12345, 'name')
|
||||||
|
expected_payload = {'filter': {'filterType': 'AND', 'filters': [
|
||||||
|
{'filterType': 'Equals', 'attributeName': 'scSerialNumber',
|
||||||
|
'attributeValue': 12345},
|
||||||
|
{'filterType': 'Equals', 'attributeName': 'name',
|
||||||
|
'attributeValue': 'name'}]}}
|
||||||
|
mock_post.assert_called_once_with('StorageCenter/ScDiskFolder/GetList',
|
||||||
|
expected_payload)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
'get',
|
'get',
|
||||||
return_value=RESPONSE_200)
|
return_value=RESPONSE_200)
|
||||||
@ -6446,6 +6401,250 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
scvol,
|
scvol,
|
||||||
'a,b')
|
'a,b')
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'get')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json')
|
||||||
|
def test_get_live_volume(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_get,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
# Basic check
|
||||||
|
ret = self.scapi.get_live_volume(None)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
lv1 = {'primaryVolume': {'instanceId': '12345.1'}}
|
||||||
|
lv2 = {'primaryVolume': {'instanceId': '12345.2'}}
|
||||||
|
mock_get_json.return_value = [lv1, lv2]
|
||||||
|
mock_get.return_value = self.RESPONSE_200
|
||||||
|
# Good Run
|
||||||
|
ret = self.scapi.get_live_volume('12345.2')
|
||||||
|
self.assertEqual(lv2, ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'get')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json')
|
||||||
|
def test_get_live_volume_not_found(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_get,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
lv1 = {'primaryVolume': {'instanceId': '12345.1'}}
|
||||||
|
lv2 = {'primaryVolume': {'instanceId': '12345.2'}}
|
||||||
|
mock_get_json.return_value = [lv1, lv2]
|
||||||
|
mock_get.return_value = self.RESPONSE_200
|
||||||
|
ret = self.scapi.get_live_volume('12345.3')
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'get')
|
||||||
|
def test_get_live_volume_error(self,
|
||||||
|
mock_get,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
mock_get.return_value = self.RESPONSE_400
|
||||||
|
ret = self.scapi.get_live_volume('12345.2')
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'post')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json')
|
||||||
|
def test_map_secondary_volume(self,
|
||||||
|
mock_get_json,
|
||||||
|
mock_post,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101'},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
scdestsrv = {'instanceId': '102.1000'}
|
||||||
|
mock_post.return_value = self.RESPONSE_200
|
||||||
|
mock_get_json.return_value = {'instanceId': '102.101.1'}
|
||||||
|
ret = self.scapi.map_secondary_volume(sclivevol, scdestsrv)
|
||||||
|
expected_payload = {'Server': '102.1000',
|
||||||
|
'Advanced': {'MapToDownServerHbas': True}}
|
||||||
|
mock_post.assert_called_once_with(
|
||||||
|
'StorageCenter/ScLiveVolume/101.101/MapSecondaryVolume',
|
||||||
|
expected_payload, True
|
||||||
|
)
|
||||||
|
self.assertEqual({'instanceId': '102.101.1'}, ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'post')
|
||||||
|
def test_map_secondary_volume_err(self,
|
||||||
|
mock_post,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101'},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
scdestsrv = {'instanceId': '102.1000'}
|
||||||
|
mock_post.return_value = self.RESPONSE_400
|
||||||
|
ret = self.scapi.map_secondary_volume(sclivevol, scdestsrv)
|
||||||
|
expected_payload = {'Server': '102.1000',
|
||||||
|
'Advanced': {'MapToDownServerHbas': True}}
|
||||||
|
mock_post.assert_called_once_with(
|
||||||
|
'StorageCenter/ScLiveVolume/101.101/MapSecondaryVolume',
|
||||||
|
expected_payload, True
|
||||||
|
)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'post')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_get_json')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_find_qos')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_sc')
|
||||||
|
def test_create_live_volume(self,
|
||||||
|
mock_find_sc,
|
||||||
|
mock_find_qos,
|
||||||
|
mock_get_json,
|
||||||
|
mock_post,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
scvol = {'instanceId': '101.1',
|
||||||
|
'name': 'name'}
|
||||||
|
sclivevol = {'instanceId': '101.101',
|
||||||
|
'secondaryVolume': {'instanceId': '102.101'},
|
||||||
|
'secondaryScSerialNumber': 102}
|
||||||
|
|
||||||
|
remotessn = '102'
|
||||||
|
active = True
|
||||||
|
sync = False
|
||||||
|
primaryqos = 'fast'
|
||||||
|
secondaryqos = 'slow'
|
||||||
|
mock_find_sc.return_value = 102
|
||||||
|
mock_find_qos.side_effect = [{'instanceId': '101.1001'},
|
||||||
|
{'instanceId': '102.1001'}]
|
||||||
|
mock_post.return_value = self.RESPONSE_200
|
||||||
|
mock_get_json.return_value = sclivevol
|
||||||
|
ret = self.scapi.create_live_volume(scvol, remotessn, active, sync,
|
||||||
|
primaryqos, secondaryqos)
|
||||||
|
mock_find_sc.assert_called_once_with(102)
|
||||||
|
mock_find_qos.assert_any_call(primaryqos)
|
||||||
|
mock_find_qos.assert_any_call(secondaryqos, 102)
|
||||||
|
self.assertEqual(sclivevol, ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'post')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_find_qos')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_sc')
|
||||||
|
def test_create_live_volume_error(self,
|
||||||
|
mock_find_sc,
|
||||||
|
mock_find_qos,
|
||||||
|
mock_post,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
scvol = {'instanceId': '101.1',
|
||||||
|
'name': 'name'}
|
||||||
|
remotessn = '102'
|
||||||
|
active = True
|
||||||
|
sync = False
|
||||||
|
primaryqos = 'fast'
|
||||||
|
secondaryqos = 'slow'
|
||||||
|
mock_find_sc.return_value = 102
|
||||||
|
mock_find_qos.side_effect = [{'instanceId': '101.1001'},
|
||||||
|
{'instanceId': '102.1001'}]
|
||||||
|
mock_post.return_value = self.RESPONSE_400
|
||||||
|
ret = self.scapi.create_live_volume(scvol, remotessn, active, sync,
|
||||||
|
primaryqos, secondaryqos)
|
||||||
|
mock_find_sc.assert_called_once_with(102)
|
||||||
|
mock_find_qos.assert_any_call(primaryqos)
|
||||||
|
mock_find_qos.assert_any_call(secondaryqos, 102)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_find_qos')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_sc')
|
||||||
|
def test_create_live_volume_no_dest(self,
|
||||||
|
mock_find_sc,
|
||||||
|
mock_find_qos,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
scvol = {'instanceId': '101.1',
|
||||||
|
'name': 'name'}
|
||||||
|
remotessn = '102'
|
||||||
|
active = True
|
||||||
|
sync = False
|
||||||
|
primaryqos = 'fast'
|
||||||
|
secondaryqos = 'slow'
|
||||||
|
mock_find_sc.return_value = 102
|
||||||
|
mock_find_qos.return_value = {}
|
||||||
|
ret = self.scapi.create_live_volume(scvol, remotessn, active, sync,
|
||||||
|
primaryqos, secondaryqos)
|
||||||
|
mock_find_sc.assert_called_once_with(102)
|
||||||
|
mock_find_qos.assert_any_call(primaryqos)
|
||||||
|
mock_find_qos.assert_any_call(secondaryqos, 102)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_find_qos')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_sc')
|
||||||
|
def test_create_live_volume_no_qos(self,
|
||||||
|
mock_find_sc,
|
||||||
|
mock_find_qos,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
scvol = {'instanceId': '101.1',
|
||||||
|
'name': 'name'}
|
||||||
|
remotessn = '102'
|
||||||
|
active = True
|
||||||
|
sync = False
|
||||||
|
primaryqos = 'fast'
|
||||||
|
secondaryqos = 'slow'
|
||||||
|
mock_find_sc.return_value = 102
|
||||||
|
mock_find_qos.return_value = None
|
||||||
|
ret = self.scapi.create_live_volume(scvol, remotessn, active, sync,
|
||||||
|
primaryqos, secondaryqos)
|
||||||
|
mock_find_sc.assert_called_once_with(102)
|
||||||
|
mock_find_qos.assert_any_call(primaryqos)
|
||||||
|
mock_find_qos.assert_any_call(secondaryqos, 102)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'_find_qos')
|
||||||
|
@mock.patch.object(dell_storagecenter_api.StorageCenterApi,
|
||||||
|
'find_sc')
|
||||||
|
def test_create_live_volume_no_secondary_qos(self,
|
||||||
|
mock_find_sc,
|
||||||
|
mock_find_qos,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
scvol = {'instanceId': '101.1',
|
||||||
|
'name': 'name'}
|
||||||
|
remotessn = '102'
|
||||||
|
active = True
|
||||||
|
sync = False
|
||||||
|
primaryqos = 'fast'
|
||||||
|
secondaryqos = 'slow'
|
||||||
|
mock_find_sc.return_value = 102
|
||||||
|
mock_find_qos.side_effect = [{'instanceId': '101.1001'},
|
||||||
|
None]
|
||||||
|
ret = self.scapi.create_live_volume(scvol, remotessn, active, sync,
|
||||||
|
primaryqos, secondaryqos)
|
||||||
|
mock_find_sc.assert_called_once_with(102)
|
||||||
|
mock_find_qos.assert_any_call(primaryqos)
|
||||||
|
mock_find_qos.assert_any_call(secondaryqos, 102)
|
||||||
|
self.assertIsNone(ret)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
'put')
|
'put')
|
||||||
def test_manage_replay(self,
|
def test_manage_replay(self,
|
||||||
@ -6707,6 +6906,22 @@ class DellSCSanAPITestCase(test.TestCase):
|
|||||||
self.assertIsNone(retbool)
|
self.assertIsNone(retbool)
|
||||||
self.assertIsNone(retnum)
|
self.assertIsNone(retnum)
|
||||||
|
|
||||||
|
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
||||||
|
'delete')
|
||||||
|
def test_delete_live_volume(self,
|
||||||
|
mock_delete,
|
||||||
|
mock_close_connection,
|
||||||
|
mock_open_connection,
|
||||||
|
mock_init):
|
||||||
|
mock_delete.return_value = self.RESPONSE_200
|
||||||
|
ret = self.scapi.delete_live_volume({'instanceId': '12345.101'},
|
||||||
|
True)
|
||||||
|
self.assertTrue(ret)
|
||||||
|
mock_delete.return_value = self.RESPONSE_400
|
||||||
|
ret = self.scapi.delete_live_volume({'instanceId': '12345.101'},
|
||||||
|
True)
|
||||||
|
self.assertFalse(ret)
|
||||||
|
|
||||||
|
|
||||||
class DellSCSanAPIConnectionTestCase(test.TestCase):
|
class DellSCSanAPIConnectionTestCase(test.TestCase):
|
||||||
|
|
||||||
@ -7009,41 +7224,18 @@ class DellHttpClientTestCase(test.TestCase):
|
|||||||
self.httpclient._wait_for_async_complete,
|
self.httpclient._wait_for_async_complete,
|
||||||
self.ASYNCTASK)
|
self.ASYNCTASK)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
|
||||||
'_rest_ret',
|
|
||||||
return_value=RESPONSE_200)
|
|
||||||
@mock.patch.object(requests.Session,
|
@mock.patch.object(requests.Session,
|
||||||
'get',
|
'get',
|
||||||
return_value=RESPONSE_200)
|
return_value=RESPONSE_200)
|
||||||
def test_get(self,
|
def test_get(self,
|
||||||
mock_get,
|
mock_get):
|
||||||
mock_rest_ret):
|
ret = self.httpclient.get('url')
|
||||||
ret = self.httpclient.get('url', False)
|
|
||||||
self.assertEqual(self.RESPONSE_200, ret)
|
self.assertEqual(self.RESPONSE_200, ret)
|
||||||
mock_rest_ret.assert_called_once_with(self.RESPONSE_200, False)
|
|
||||||
expected_headers = self.httpclient.header.copy()
|
expected_headers = self.httpclient.header.copy()
|
||||||
mock_get.assert_called_once_with('https://localhost:3033/api/rest/url',
|
mock_get.assert_called_once_with('https://localhost:3033/api/rest/url',
|
||||||
headers=expected_headers,
|
headers=expected_headers,
|
||||||
verify=False)
|
verify=False)
|
||||||
|
|
||||||
@mock.patch.object(dell_storagecenter_api.HttpClient,
|
|
||||||
'_rest_ret',
|
|
||||||
return_value=RESPONSE_200)
|
|
||||||
@mock.patch.object(requests.Session,
|
|
||||||
'get',
|
|
||||||
return_value=RESPONSE_200)
|
|
||||||
def test_get_async(self,
|
|
||||||
mock_get,
|
|
||||||
mock_rest_ret):
|
|
||||||
ret = self.httpclient.get('url', True)
|
|
||||||
self.assertEqual(self.RESPONSE_200, ret)
|
|
||||||
mock_rest_ret.assert_called_once_with(self.RESPONSE_200, True)
|
|
||||||
expected_headers = self.httpclient.header.copy()
|
|
||||||
expected_headers['async'] = True
|
|
||||||
mock_get.assert_called_once_with('https://localhost:3033/api/rest/url',
|
|
||||||
headers=expected_headers,
|
|
||||||
verify=False)
|
|
||||||
|
|
||||||
|
|
||||||
class DellStorageCenterApiHelperTestCase(test.TestCase):
|
class DellStorageCenterApiHelperTestCase(test.TestCase):
|
||||||
|
|
||||||
|
@ -200,11 +200,12 @@ class HttpClient(object):
|
|||||||
|
|
||||||
@utils.retry(exceptions=(requests.ConnectionError,
|
@utils.retry(exceptions=(requests.ConnectionError,
|
||||||
exception.DellDriverRetryableException))
|
exception.DellDriverRetryableException))
|
||||||
def get(self, url, async=False):
|
def get(self, url):
|
||||||
LOG.debug('get: %(url)s', {'url': url})
|
LOG.debug('get: %(url)s', {'url': url})
|
||||||
rest_response = self._rest_ret(self.session.get(
|
rest_response = self.session.get(self.__formatUrl(url),
|
||||||
self.__formatUrl(url), headers=self._get_header(async),
|
headers=self.header,
|
||||||
verify=self.verify), async)
|
verify=self.verify)
|
||||||
|
|
||||||
if rest_response and rest_response.status_code == 400 and (
|
if rest_response and rest_response.status_code == 400 and (
|
||||||
'Unhandled Exception' in rest_response.text):
|
'Unhandled Exception' in rest_response.text):
|
||||||
raise exception.DellDriverRetryableException()
|
raise exception.DellDriverRetryableException()
|
||||||
@ -381,7 +382,8 @@ class StorageCenterApi(object):
|
|||||||
2.4.1 - Updated Replication support to V2.1.
|
2.4.1 - Updated Replication support to V2.1.
|
||||||
2.5.0 - ManageableSnapshotsVD implemented.
|
2.5.0 - ManageableSnapshotsVD implemented.
|
||||||
3.0.0 - ProviderID utilized.
|
3.0.0 - ProviderID utilized.
|
||||||
3.1.0 - Failback Supported.
|
3.1.0 - Failback supported.
|
||||||
|
3.2.0 - Live Volume support.
|
||||||
3.3.0 - Support for a secondary DSM.
|
3.3.0 - Support for a secondary DSM.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -664,8 +666,7 @@ class StorageCenterApi(object):
|
|||||||
"""
|
"""
|
||||||
# We might be looking for another ssn. If not then
|
# We might be looking for another ssn. If not then
|
||||||
# look for our default.
|
# look for our default.
|
||||||
if ssn == -1:
|
ssn = self._vet_ssn(ssn)
|
||||||
ssn = self.ssn
|
|
||||||
|
|
||||||
r = self.client.get('StorageCenter/StorageCenter')
|
r = self.client.get('StorageCenter/StorageCenter')
|
||||||
result = self._get_result(r, 'scSerialNumber', ssn)
|
result = self._get_result(r, 'scSerialNumber', ssn)
|
||||||
@ -680,7 +681,7 @@ class StorageCenterApi(object):
|
|||||||
|
|
||||||
# Folder functions
|
# Folder functions
|
||||||
|
|
||||||
def _create_folder(self, url, parent, folder):
|
def _create_folder(self, url, parent, folder, ssn=-1):
|
||||||
"""Creates folder under parent.
|
"""Creates folder under parent.
|
||||||
|
|
||||||
This can create both to server and volume folders. The REST url
|
This can create both to server and volume folders. The REST url
|
||||||
@ -693,10 +694,12 @@ class StorageCenterApi(object):
|
|||||||
:param folder: The folder name to be created. This is one level deep.
|
:param folder: The folder name to be created. This is one level deep.
|
||||||
:returns: The REST folder object.
|
:returns: The REST folder object.
|
||||||
"""
|
"""
|
||||||
|
ssn = self._vet_ssn(ssn)
|
||||||
|
|
||||||
scfolder = None
|
scfolder = None
|
||||||
payload = {}
|
payload = {}
|
||||||
payload['Name'] = folder
|
payload['Name'] = folder
|
||||||
payload['StorageCenter'] = self.ssn
|
payload['StorageCenter'] = ssn
|
||||||
if parent != '':
|
if parent != '':
|
||||||
payload['Parent'] = parent
|
payload['Parent'] = parent
|
||||||
payload['Notes'] = self.notes
|
payload['Notes'] = self.notes
|
||||||
@ -706,7 +709,7 @@ class StorageCenterApi(object):
|
|||||||
scfolder = self._first_result(r)
|
scfolder = self._first_result(r)
|
||||||
return scfolder
|
return scfolder
|
||||||
|
|
||||||
def _create_folder_path(self, url, foldername):
|
def _create_folder_path(self, url, foldername, ssn=-1):
|
||||||
"""Creates a folder path from a fully qualified name.
|
"""Creates a folder path from a fully qualified name.
|
||||||
|
|
||||||
The REST url sent in defines the folder type being created on the Dell
|
The REST url sent in defines the folder type being created on the Dell
|
||||||
@ -718,6 +721,8 @@ class StorageCenterApi(object):
|
|||||||
:param foldername: The full folder name with path.
|
:param foldername: The full folder name with path.
|
||||||
:returns: The REST folder object.
|
:returns: The REST folder object.
|
||||||
"""
|
"""
|
||||||
|
ssn = self._vet_ssn(ssn)
|
||||||
|
|
||||||
path = self._path_to_array(foldername)
|
path = self._path_to_array(foldername)
|
||||||
folderpath = ''
|
folderpath = ''
|
||||||
instanceId = ''
|
instanceId = ''
|
||||||
@ -729,12 +734,12 @@ class StorageCenterApi(object):
|
|||||||
# If the last was found see if this part of the path exists too
|
# If the last was found see if this part of the path exists too
|
||||||
if found:
|
if found:
|
||||||
listurl = url + '/GetList'
|
listurl = url + '/GetList'
|
||||||
scfolder = self._find_folder(listurl, folderpath)
|
scfolder = self._find_folder(listurl, folderpath, ssn)
|
||||||
if scfolder is None:
|
if scfolder is None:
|
||||||
found = False
|
found = False
|
||||||
# We didn't find it so create it
|
# We didn't find it so create it
|
||||||
if found is False:
|
if found is False:
|
||||||
scfolder = self._create_folder(url, instanceId, folder)
|
scfolder = self._create_folder(url, instanceId, folder, ssn)
|
||||||
# If we haven't found a folder or created it then leave
|
# If we haven't found a folder or created it then leave
|
||||||
if scfolder is None:
|
if scfolder is None:
|
||||||
LOG.error(_LE('Unable to create folder path %s'), folderpath)
|
LOG.error(_LE('Unable to create folder path %s'), folderpath)
|
||||||
@ -744,7 +749,7 @@ class StorageCenterApi(object):
|
|||||||
folderpath = folderpath + '/'
|
folderpath = folderpath + '/'
|
||||||
return scfolder
|
return scfolder
|
||||||
|
|
||||||
def _find_folder(self, url, foldername):
|
def _find_folder(self, url, foldername, ssn=-1):
|
||||||
"""Find a folder on the SC using the specified url.
|
"""Find a folder on the SC using the specified url.
|
||||||
|
|
||||||
Most of the time the folder will already have been created so
|
Most of the time the folder will already have been created so
|
||||||
@ -761,8 +766,10 @@ class StorageCenterApi(object):
|
|||||||
:param foldername: Full path to the folder we are looking for.
|
:param foldername: Full path to the folder we are looking for.
|
||||||
:returns: Dell folder object.
|
:returns: Dell folder object.
|
||||||
"""
|
"""
|
||||||
|
ssn = self._vet_ssn(ssn)
|
||||||
|
|
||||||
pf = self._get_payload_filter()
|
pf = self._get_payload_filter()
|
||||||
pf.append('scSerialNumber', self.ssn)
|
pf.append('scSerialNumber', ssn)
|
||||||
basename = os.path.basename(foldername)
|
basename = os.path.basename(foldername)
|
||||||
pf.append('Name', basename)
|
pf.append('Name', basename)
|
||||||
# If we have any kind of path we throw it into the filters.
|
# If we have any kind of path we throw it into the filters.
|
||||||
@ -777,7 +784,7 @@ class StorageCenterApi(object):
|
|||||||
folder = self._get_result(r, 'folderPath', folderpath)
|
folder = self._get_result(r, 'folderPath', folderpath)
|
||||||
return folder
|
return folder
|
||||||
|
|
||||||
def _find_volume_folder(self, create=False):
|
def _find_volume_folder(self, create=False, ssn=-1):
|
||||||
"""Looks for the volume folder where backend volumes will be created.
|
"""Looks for the volume folder where backend volumes will be created.
|
||||||
|
|
||||||
Volume folder is specified in the cindef.conf. See __init.
|
Volume folder is specified in the cindef.conf. See __init.
|
||||||
@ -786,11 +793,11 @@ class StorageCenterApi(object):
|
|||||||
:returns: Folder object.
|
:returns: Folder object.
|
||||||
"""
|
"""
|
||||||
folder = self._find_folder('StorageCenter/ScVolumeFolder/GetList',
|
folder = self._find_folder('StorageCenter/ScVolumeFolder/GetList',
|
||||||
self.vfname)
|
self.vfname, ssn)
|
||||||
# Doesn't exist? make it
|
# Doesn't exist? make it
|
||||||
if folder is None and create is True:
|
if folder is None and create is True:
|
||||||
folder = self._create_folder_path('StorageCenter/ScVolumeFolder',
|
folder = self._create_folder_path('StorageCenter/ScVolumeFolder',
|
||||||
self.vfname)
|
self.vfname, ssn)
|
||||||
return folder
|
return folder
|
||||||
|
|
||||||
def _init_volume(self, scvolume):
|
def _init_volume(self, scvolume):
|
||||||
@ -1051,8 +1058,7 @@ class StorageCenterApi(object):
|
|||||||
:param ssn: SSN to search on.
|
:param ssn: SSN to search on.
|
||||||
:return: Returns the scvolume list or None.
|
:return: Returns the scvolume list or None.
|
||||||
"""
|
"""
|
||||||
if ssn == -1:
|
ssn = self._vet_ssn(ssn)
|
||||||
ssn = self.ssn
|
|
||||||
result = None
|
result = None
|
||||||
# We need a name or a device ID to find a volume.
|
# We need a name or a device ID to find a volume.
|
||||||
if name or deviceid:
|
if name or deviceid:
|
||||||
@ -1194,7 +1200,7 @@ class StorageCenterApi(object):
|
|||||||
'provider_id: %s'), provider_id)
|
'provider_id: %s'), provider_id)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _find_server_folder(self, create=False):
|
def _find_server_folder(self, create=False, ssn=-1):
|
||||||
"""Looks for the server folder on the Dell Storage Center.
|
"""Looks for the server folder on the Dell Storage Center.
|
||||||
|
|
||||||
This is the folder where a server objects for mapping volumes will be
|
This is the folder where a server objects for mapping volumes will be
|
||||||
@ -1203,11 +1209,13 @@ class StorageCenterApi(object):
|
|||||||
:param create: If True will create the folder if not found.
|
:param create: If True will create the folder if not found.
|
||||||
:return: Folder object.
|
:return: Folder object.
|
||||||
"""
|
"""
|
||||||
|
ssn = self._vet_ssn(ssn)
|
||||||
|
|
||||||
folder = self._find_folder('StorageCenter/ScServerFolder/GetList',
|
folder = self._find_folder('StorageCenter/ScServerFolder/GetList',
|
||||||
self.sfname)
|
self.sfname, ssn)
|
||||||
if folder is None and create is True:
|
if folder is None and create is True:
|
||||||
folder = self._create_folder_path('StorageCenter/ScServerFolder',
|
folder = self._create_folder_path('StorageCenter/ScServerFolder',
|
||||||
self.sfname)
|
self.sfname, ssn)
|
||||||
return folder
|
return folder
|
||||||
|
|
||||||
def _add_hba(self, scserver, wwnoriscsiname):
|
def _add_hba(self, scserver, wwnoriscsiname):
|
||||||
@ -1235,7 +1243,7 @@ class StorageCenterApi(object):
|
|||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _find_serveros(self, osname='Red Hat Linux 6.x'):
|
def _find_serveros(self, osname='Red Hat Linux 6.x', ssn=-1):
|
||||||
"""Returns the serveros instance id of the specified osname.
|
"""Returns the serveros instance id of the specified osname.
|
||||||
|
|
||||||
Required to create a Dell server object.
|
Required to create a Dell server object.
|
||||||
@ -1246,8 +1254,9 @@ class StorageCenterApi(object):
|
|||||||
:param osname: The name of the OS to look for.
|
:param osname: The name of the OS to look for.
|
||||||
:returns: InstanceId of the ScServerOperatingSystem object.
|
:returns: InstanceId of the ScServerOperatingSystem object.
|
||||||
"""
|
"""
|
||||||
|
ssn = self._vet_ssn(ssn)
|
||||||
pf = self._get_payload_filter()
|
pf = self._get_payload_filter()
|
||||||
pf.append('scSerialNumber', self.ssn)
|
pf.append('scSerialNumber', ssn)
|
||||||
r = self.client.post('StorageCenter/ScServerOperatingSystem/GetList',
|
r = self.client.post('StorageCenter/ScServerOperatingSystem/GetList',
|
||||||
pf.payload)
|
pf.payload)
|
||||||
if self._check_result(r):
|
if self._check_result(r):
|
||||||
@ -1262,55 +1271,47 @@ class StorageCenterApi(object):
|
|||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def create_server_multiple_hbas(self, wwns):
|
def create_server(self, wwnlist, ssn=-1):
|
||||||
"""Creates a server with multiple WWNS associated with it.
|
"""Creates a server with multiple WWNS associated with it.
|
||||||
|
|
||||||
Same as create_server except it can take a list of HBAs.
|
Same as create_server except it can take a list of HBAs.
|
||||||
|
|
||||||
:param wwns: A list of FC WWNs or iSCSI IQNs associated with this
|
:param wwnlist: A list of FC WWNs or iSCSI IQNs associated with this
|
||||||
server.
|
server.
|
||||||
:returns: Dell server object.
|
:returns: Dell server object.
|
||||||
"""
|
"""
|
||||||
scserver = None
|
# Find our folder or make it
|
||||||
# Our instance names
|
folder = self._find_server_folder(True, ssn)
|
||||||
for wwn in wwns:
|
# Create our server.
|
||||||
if scserver is None:
|
scserver = self._create_server('Server_' + wwnlist[0], folder, ssn)
|
||||||
# Use the fist wwn to create the server.
|
if not scserver:
|
||||||
scserver = self.create_server(wwn)
|
return None
|
||||||
else:
|
# Add our HBAs.
|
||||||
# Add the wwn to our server
|
if scserver:
|
||||||
self._add_hba(scserver, wwn)
|
for wwn in wwnlist:
|
||||||
|
if not self._add_hba(scserver, wwn):
|
||||||
|
# We failed so log it. Delete our server and return None.
|
||||||
|
LOG.error(_LE('Error adding HBA %s to server'), wwn)
|
||||||
|
self._delete_server(scserver)
|
||||||
|
return None
|
||||||
return scserver
|
return scserver
|
||||||
|
|
||||||
def create_server(self, wwnoriscsiname):
|
def _create_server(self, servername, folder, ssn):
|
||||||
"""Creates a Dell server object on the the Storage Center.
|
ssn = self._vet_ssn(ssn)
|
||||||
|
|
||||||
Adds the first HBA identified by wwnoriscsiname to it.
|
LOG.info(_LI('Creating server %s'), servername)
|
||||||
|
|
||||||
:param wwnoriscsiname: FC WWN or iSCSI IQN associated with
|
|
||||||
this Dell server object.
|
|
||||||
:returns: Dell server object.
|
|
||||||
"""
|
|
||||||
|
|
||||||
LOG.info(_LI('Creating server %s'), wwnoriscsiname)
|
|
||||||
|
|
||||||
scserver = None
|
|
||||||
payload = {}
|
payload = {}
|
||||||
payload['Name'] = 'Server_' + wwnoriscsiname
|
payload['Name'] = servername
|
||||||
payload['StorageCenter'] = self.ssn
|
payload['StorageCenter'] = ssn
|
||||||
payload['Notes'] = self.notes
|
payload['Notes'] = self.notes
|
||||||
# We pick Red Hat Linux 6.x because it supports multipath and
|
# We pick Red Hat Linux 6.x because it supports multipath and
|
||||||
# will attach luns to paths as they are found.
|
# will attach luns to paths as they are found.
|
||||||
scserveros = self._find_serveros('Red Hat Linux 6.x')
|
scserveros = self._find_serveros('Red Hat Linux 6.x', ssn)
|
||||||
if scserveros is not None:
|
if scserveros is not None:
|
||||||
payload['OperatingSystem'] = scserveros
|
payload['OperatingSystem'] = scserveros
|
||||||
|
|
||||||
# Find our folder or make it
|
# At this point it doesn't matter if we have a folder or not.
|
||||||
folder = self._find_server_folder(True)
|
# Let it be in the root if the folder creation fails.
|
||||||
|
|
||||||
# At this point it doesn't matter if the folder was created or not.
|
|
||||||
# We just attempt to create the server. Let it be in the root if
|
|
||||||
# the folder creation fails.
|
|
||||||
if folder is not None:
|
if folder is not None:
|
||||||
payload['ServerFolder'] = self._get_id(folder)
|
payload['ServerFolder'] = self._get_id(folder)
|
||||||
|
|
||||||
@ -1320,19 +1321,24 @@ class StorageCenterApi(object):
|
|||||||
# Server was created
|
# Server was created
|
||||||
scserver = self._first_result(r)
|
scserver = self._first_result(r)
|
||||||
LOG.info(_LI('SC server created %s'), scserver)
|
LOG.info(_LI('SC server created %s'), scserver)
|
||||||
|
return scserver
|
||||||
|
LOG.error(_LE('Unable to create SC server %s'), servername)
|
||||||
|
return None
|
||||||
|
|
||||||
# Add hba to our server
|
def _vet_ssn(self, ssn):
|
||||||
if scserver is not None:
|
"""Returns the default if a ssn was not set.
|
||||||
if not self._add_hba(scserver, wwnoriscsiname):
|
|
||||||
LOG.error(_LE('Error adding HBA to server'))
|
|
||||||
# Can't have a server without an HBA
|
|
||||||
self._delete_server(scserver)
|
|
||||||
scserver = None
|
|
||||||
|
|
||||||
# Success or failure is determined by the caller
|
Added to support live volume as we aren't always on the primary ssn
|
||||||
return scserver
|
anymore
|
||||||
|
|
||||||
def find_server(self, instance_name):
|
:param ssn: ssn to check.
|
||||||
|
:return: Current ssn or the ssn sent down.
|
||||||
|
"""
|
||||||
|
if ssn == -1:
|
||||||
|
return self.ssn
|
||||||
|
return ssn
|
||||||
|
|
||||||
|
def find_server(self, instance_name, ssn=-1):
|
||||||
"""Hunts for a server on the Dell backend by instance_name.
|
"""Hunts for a server on the Dell backend by instance_name.
|
||||||
|
|
||||||
The instance_name is the same as the server's HBA. This is the IQN or
|
The instance_name is the same as the server's HBA. This is the IQN or
|
||||||
@ -1342,17 +1348,20 @@ class StorageCenterApi(object):
|
|||||||
:param instance_name: instance_name is a FC WWN or iSCSI IQN from
|
:param instance_name: instance_name is a FC WWN or iSCSI IQN from
|
||||||
the connector. In cinder a server is identified
|
the connector. In cinder a server is identified
|
||||||
by its HBA.
|
by its HBA.
|
||||||
|
:param ssn: Storage center to search.
|
||||||
:returns: Dell server object or None.
|
:returns: Dell server object or None.
|
||||||
"""
|
"""
|
||||||
|
ssn = self._vet_ssn(ssn)
|
||||||
|
|
||||||
scserver = None
|
scserver = None
|
||||||
# We search for our server by first finding our HBA
|
# We search for our server by first finding our HBA
|
||||||
hba = self._find_serverhba(instance_name)
|
hba = self._find_serverhba(instance_name, ssn)
|
||||||
# Once created hbas stay in the system. So it isn't enough
|
# Once created hbas stay in the system. So it isn't enough
|
||||||
# that we found one it actually has to be attached to a
|
# that we found one it actually has to be attached to a
|
||||||
# server.
|
# server.
|
||||||
if hba is not None and hba.get('server') is not None:
|
if hba is not None and hba.get('server') is not None:
|
||||||
pf = self._get_payload_filter()
|
pf = self._get_payload_filter()
|
||||||
pf.append('scSerialNumber', self.ssn)
|
pf.append('scSerialNumber', ssn)
|
||||||
pf.append('instanceId', self._get_id(hba['server']))
|
pf.append('instanceId', self._get_id(hba['server']))
|
||||||
r = self.client.post('StorageCenter/ScServer/GetList', pf.payload)
|
r = self.client.post('StorageCenter/ScServer/GetList', pf.payload)
|
||||||
if self._check_result(r):
|
if self._check_result(r):
|
||||||
@ -1362,7 +1371,7 @@ class StorageCenterApi(object):
|
|||||||
LOG.debug('Server (%s) not found.', instance_name)
|
LOG.debug('Server (%s) not found.', instance_name)
|
||||||
return scserver
|
return scserver
|
||||||
|
|
||||||
def _find_serverhba(self, instance_name):
|
def _find_serverhba(self, instance_name, ssn):
|
||||||
"""Hunts for a server HBA on the Dell backend by instance_name.
|
"""Hunts for a server HBA on the Dell backend by instance_name.
|
||||||
|
|
||||||
Instance_name is the same as the IQN or WWN specified in the
|
Instance_name is the same as the IQN or WWN specified in the
|
||||||
@ -1370,12 +1379,13 @@ class StorageCenterApi(object):
|
|||||||
|
|
||||||
:param instance_name: Instance_name is a FC WWN or iSCSI IQN from
|
:param instance_name: Instance_name is a FC WWN or iSCSI IQN from
|
||||||
the connector.
|
the connector.
|
||||||
|
:param ssn: Storage center to search.
|
||||||
:returns: Dell server HBA object.
|
:returns: Dell server HBA object.
|
||||||
"""
|
"""
|
||||||
scserverhba = None
|
scserverhba = None
|
||||||
# We search for our server by first finding our HBA
|
# We search for our server by first finding our HBA
|
||||||
pf = self._get_payload_filter()
|
pf = self._get_payload_filter()
|
||||||
pf.append('scSerialNumber', self.ssn)
|
pf.append('scSerialNumber', ssn)
|
||||||
pf.append('instanceName', instance_name)
|
pf.append('instanceName', instance_name)
|
||||||
r = self.client.post('StorageCenter/ScServerHba/GetList', pf.payload)
|
r = self.client.post('StorageCenter/ScServerHba/GetList', pf.payload)
|
||||||
if self._check_result(r):
|
if self._check_result(r):
|
||||||
@ -1449,6 +1459,7 @@ class StorageCenterApi(object):
|
|||||||
LOG.info(_LI('Volume mappings for %(name)s: %(mappings)s'),
|
LOG.info(_LI('Volume mappings for %(name)s: %(mappings)s'),
|
||||||
{'name': scvolume.get('name'),
|
{'name': scvolume.get('name'),
|
||||||
'mappings': mappings})
|
'mappings': mappings})
|
||||||
|
|
||||||
return mappings
|
return mappings
|
||||||
|
|
||||||
def _find_mapping_profiles(self, scvolume):
|
def _find_mapping_profiles(self, scvolume):
|
||||||
@ -1599,23 +1610,19 @@ class StorageCenterApi(object):
|
|||||||
'Error finding configuration: %s'), cportid)
|
'Error finding configuration: %s'), cportid)
|
||||||
return controllerport
|
return controllerport
|
||||||
|
|
||||||
def find_iscsi_properties(self, scvolume, ip=None, port=None):
|
def find_iscsi_properties(self, scvolume):
|
||||||
"""Finds target information for a given Dell scvolume object mapping.
|
"""Finds target information for a given Dell scvolume object mapping.
|
||||||
|
|
||||||
The data coming back is both the preferred path and all the paths.
|
The data coming back is both the preferred path and all the paths.
|
||||||
|
|
||||||
:param scvolume: The dell sc volume object.
|
:param scvolume: The dell sc volume object.
|
||||||
:param ip: The preferred target portal ip.
|
|
||||||
:param port: The preferred target portal port.
|
|
||||||
:returns: iSCSI property dictionary.
|
:returns: iSCSI property dictionary.
|
||||||
:raises: VolumeBackendAPIException
|
:raises: VolumeBackendAPIException
|
||||||
"""
|
"""
|
||||||
LOG.debug('find_iscsi_properties: scvolume: %s', scvolume)
|
LOG.debug('find_iscsi_properties: scvolume: %s', scvolume)
|
||||||
# Our mutable process object.
|
# Our mutable process object.
|
||||||
pdata = {'active': -1,
|
pdata = {'active': -1,
|
||||||
'up': -1,
|
'up': -1}
|
||||||
'ip': ip,
|
|
||||||
'port': port}
|
|
||||||
# Our output lists.
|
# Our output lists.
|
||||||
portals = []
|
portals = []
|
||||||
luns = []
|
luns = []
|
||||||
@ -1640,22 +1647,15 @@ class StorageCenterApi(object):
|
|||||||
iqns.append(iqn)
|
iqns.append(iqn)
|
||||||
luns.append(lun)
|
luns.append(lun)
|
||||||
|
|
||||||
# We've all the information. We need to find
|
# We need to point to the best link.
|
||||||
# the best single portal to return. So check
|
# So state active and status up is preferred
|
||||||
# this one if it is on the right IP, port and
|
# but we don't actually need the state to be
|
||||||
# if the access and status are correct.
|
# up at this point.
|
||||||
if ((pdata['ip'] is None or pdata['ip'] == address) and
|
if pdata['up'] == -1:
|
||||||
(pdata['port'] is None or pdata['port'] == port)):
|
if active:
|
||||||
|
pdata['active'] = len(iqns) - 1
|
||||||
# We need to point to the best link.
|
if status == 'Up':
|
||||||
# So state active and status up is preferred
|
pdata['up'] = pdata['active']
|
||||||
# but we don't actually need the state to be
|
|
||||||
# up at this point.
|
|
||||||
if pdata['up'] == -1:
|
|
||||||
if active:
|
|
||||||
pdata['active'] = len(iqns) - 1
|
|
||||||
if status == 'Up':
|
|
||||||
pdata['up'] = pdata['active']
|
|
||||||
|
|
||||||
# Start by getting our mappings.
|
# Start by getting our mappings.
|
||||||
mappings = self._find_mappings(scvolume)
|
mappings = self._find_mappings(scvolume)
|
||||||
@ -1672,6 +1672,11 @@ class StorageCenterApi(object):
|
|||||||
isvpmode = self._is_virtualport_mode()
|
isvpmode = self._is_virtualport_mode()
|
||||||
# Trundle through our mappings.
|
# Trundle through our mappings.
|
||||||
for mapping in mappings:
|
for mapping in mappings:
|
||||||
|
# Don't return remote sc links.
|
||||||
|
msrv = mapping.get('server')
|
||||||
|
if msrv and msrv.get('objectType') == 'ScRemoteStorageCenter':
|
||||||
|
continue
|
||||||
|
|
||||||
# The lun, ro mode and status are in the mapping.
|
# The lun, ro mode and status are in the mapping.
|
||||||
LOG.debug('find_iscsi_properties: mapping: %s', mapping)
|
LOG.debug('find_iscsi_properties: mapping: %s', mapping)
|
||||||
lun = mapping.get('lun')
|
lun = mapping.get('lun')
|
||||||
@ -2605,8 +2610,7 @@ class StorageCenterApi(object):
|
|||||||
:param ssn: SSN to search on.
|
:param ssn: SSN to search on.
|
||||||
:return: scqos node object.
|
:return: scqos node object.
|
||||||
"""
|
"""
|
||||||
if ssn == -1:
|
ssn = self._vet_ssn(ssn)
|
||||||
ssn = self.ssn
|
|
||||||
pf = self._get_payload_filter()
|
pf = self._get_payload_filter()
|
||||||
pf.append('scSerialNumber', ssn)
|
pf.append('scSerialNumber', ssn)
|
||||||
pf.append('name', qosnode)
|
pf.append('name', qosnode)
|
||||||
@ -3002,3 +3006,128 @@ class StorageCenterApi(object):
|
|||||||
' progress information returned: %s'),
|
' progress information returned: %s'),
|
||||||
progress)
|
progress)
|
||||||
return None, None
|
return None, None
|
||||||
|
|
||||||
|
def get_live_volume(self, primaryid):
|
||||||
|
"""Get's the live ScLiveVolume object for the vol with primaryid.
|
||||||
|
|
||||||
|
:param primaryid: InstanceId of the primary volume.
|
||||||
|
:return: ScLiveVolume object or None.
|
||||||
|
"""
|
||||||
|
if primaryid:
|
||||||
|
r = self.client.get('StorageCenter/ScLiveVolume')
|
||||||
|
if self._check_result(r):
|
||||||
|
lvs = self._get_json(r)
|
||||||
|
for lv in lvs:
|
||||||
|
if lv['primaryVolume']['instanceId'] == primaryid:
|
||||||
|
return lv
|
||||||
|
return None
|
||||||
|
|
||||||
|
def _get_hbas(self, serverid):
|
||||||
|
# Helper to get the hba's of a given server.
|
||||||
|
r = self.client.get('StorageCenter/ScServer/%s/HbaList' % serverid)
|
||||||
|
if self._check_result(r):
|
||||||
|
return self._get_json(r)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def map_secondary_volume(self, sclivevol, scdestsrv):
|
||||||
|
"""Map's the secondary volume or a LiveVolume to destsrv.
|
||||||
|
|
||||||
|
:param sclivevol: ScLiveVolume object.
|
||||||
|
:param scdestsrv: ScServer object for the destination.
|
||||||
|
:return: ScMappingProfile object or None on failure.
|
||||||
|
"""
|
||||||
|
payload = {}
|
||||||
|
payload['Server'] = self._get_id(scdestsrv)
|
||||||
|
payload['Advanced'] = {'MapToDownServerHbas': True}
|
||||||
|
r = self.client.post('StorageCenter/ScLiveVolume/%s/MapSecondaryVolume'
|
||||||
|
% self._get_id(sclivevol), payload, True)
|
||||||
|
if self._check_result(r):
|
||||||
|
return self._get_json(r)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def create_live_volume(self, scvolume, remotessn, active=False, sync=False,
|
||||||
|
primaryqos='CinderQOS', secondaryqos='CinderQOS'):
|
||||||
|
"""This create's a live volume instead of a replication.
|
||||||
|
|
||||||
|
Servers are not created at this point so we cannot map up a remote
|
||||||
|
server immediately.
|
||||||
|
|
||||||
|
:param scvolume: Source SC Volume
|
||||||
|
:param remotessn: Destination SSN.
|
||||||
|
:param active: Replicate the active replay boolean.
|
||||||
|
:param sync: Sync replication boolean.
|
||||||
|
:param primaryqos: QOS node name for the primary side.
|
||||||
|
:param secondaryqos: QOS node name for the remote side.
|
||||||
|
:return: ScLiveVolume object or None on failure.
|
||||||
|
"""
|
||||||
|
destssn = self.find_sc(int(remotessn))
|
||||||
|
pscqos = self._find_qos(primaryqos)
|
||||||
|
sscqos = self._find_qos(secondaryqos, destssn)
|
||||||
|
if not destssn:
|
||||||
|
LOG.error(_LE('create_live_volume: Unable to find remote %s'),
|
||||||
|
remotessn)
|
||||||
|
elif not pscqos:
|
||||||
|
LOG.error(_LE('create_live_volume: Unable to find or create '
|
||||||
|
'qos node %s'), primaryqos)
|
||||||
|
elif not sscqos:
|
||||||
|
LOG.error(_LE('create_live_volume: Unable to find or create remote'
|
||||||
|
' qos node %(qos)s on %(ssn)s'),
|
||||||
|
{'qos': secondaryqos, 'ssn': destssn})
|
||||||
|
else:
|
||||||
|
payload = {}
|
||||||
|
payload['PrimaryVolume'] = self._get_id(scvolume)
|
||||||
|
payload['PrimaryQosNode'] = self._get_id(pscqos)
|
||||||
|
payload['SecondaryQosNode'] = self._get_id(sscqos)
|
||||||
|
payload['SecondaryStorageCenter'] = destssn
|
||||||
|
payload['StorageCenter'] = self.ssn
|
||||||
|
# payload['Dedup'] = False
|
||||||
|
payload['FailoverAutomaticallyEnabled'] = False
|
||||||
|
payload['RestoreAutomaticallyEnabled'] = False
|
||||||
|
payload['SwapRolesAutomaticallyEnabled'] = False
|
||||||
|
payload['ReplicateActiveReplay'] = active
|
||||||
|
payload['Type'] = 'Synchronous' if sync else 'Asynchronous'
|
||||||
|
secondaryvolumeattributes = {}
|
||||||
|
secondaryvolumeattributes['CreateSourceVolumeFolderPath'] = True
|
||||||
|
secondaryvolumeattributes['Notes'] = self.notes
|
||||||
|
secondaryvolumeattributes['Name'] = self._repl_name(
|
||||||
|
scvolume['name'])
|
||||||
|
payload[
|
||||||
|
'SecondaryVolumeAttributes'] = secondaryvolumeattributes
|
||||||
|
|
||||||
|
r = self.client.post('StorageCenter/ScLiveVolume', payload, True)
|
||||||
|
if self._check_result(r):
|
||||||
|
LOG.info(_LI('create_live_volume: Live Volume created from'
|
||||||
|
'%(svol)s to %(ssn)s'),
|
||||||
|
{'svol': self._get_id(scvolume), 'ssn': remotessn})
|
||||||
|
return self._get_json(r)
|
||||||
|
LOG.error(_LE('create_live_volume: Failed to create Live Volume from'
|
||||||
|
'%(svol)s to %(ssn)s'),
|
||||||
|
{'svol': self._get_id(scvolume), 'ssn': remotessn})
|
||||||
|
return None
|
||||||
|
|
||||||
|
def delete_live_volume(self, sclivevolume, deletesecondaryvolume):
|
||||||
|
"""Deletes the live volume.
|
||||||
|
|
||||||
|
:param sclivevolume: ScLiveVolume object to be whacked.
|
||||||
|
:return: Boolean on success/fail.
|
||||||
|
"""
|
||||||
|
payload = {}
|
||||||
|
payload['DeleteSecondaryVolume'] = deletesecondaryvolume
|
||||||
|
payload['RecycleSecondaryVolume'] = False
|
||||||
|
r = self.client.delete('StorageCenter/ScLiveVolume/%s' %
|
||||||
|
self._get_id(sclivevolume), payload, True)
|
||||||
|
if self._check_result(r):
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
def swap_roles_live_volume(self, sclivevolume):
|
||||||
|
"""Swap live volume roles.
|
||||||
|
|
||||||
|
:param sclivevolume: Dell SC live volume object.
|
||||||
|
:return: True/False on success/failure.
|
||||||
|
"""
|
||||||
|
r = self.client.post('StorageCenter/ScLiveVolume/%s/SwapRoles' %
|
||||||
|
self._get_id(sclivevolume), {}, True)
|
||||||
|
if self._check_result(r):
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
@ -157,24 +157,55 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
if profile:
|
if profile:
|
||||||
api.update_cg_volumes(profile, [volume])
|
api.update_cg_volumes(profile, [volume])
|
||||||
|
|
||||||
def _do_repl(self, api, volume):
|
def _get_replication_specs(self, volume):
|
||||||
"""Checks if we can do replication.
|
"""Checks if we can do replication.
|
||||||
|
|
||||||
Need the extra spec set and we have to be talking to EM.
|
Need the extra spec set and we have to be talking to EM.
|
||||||
|
|
||||||
:param api: Dell REST API object.
|
|
||||||
:param volume: Cinder Volume object.
|
:param volume: Cinder Volume object.
|
||||||
:return: Boolean (True if replication enabled), Boolean (True if
|
:return: rinfo dict.
|
||||||
replication type is sync.
|
|
||||||
"""
|
"""
|
||||||
do_repl = False
|
rinfo = {'enabled': False, 'sync': False,
|
||||||
sync = False
|
'live': False, 'active': False}
|
||||||
# Repl does not work with direct connect.
|
# Repl does not work with direct connect.
|
||||||
if not self.failed_over and not self.is_direct_connect:
|
if not self.is_direct_connect:
|
||||||
specs = self._get_volume_extra_specs(volume)
|
specs = self._get_volume_extra_specs(volume)
|
||||||
do_repl = specs.get('replication_enabled') == '<is> True'
|
if (not self.failed_over and
|
||||||
sync = specs.get('replication_type') == '<in> sync'
|
specs.get('replication_enabled') == '<is> True'):
|
||||||
return do_repl, sync
|
rinfo['enabled'] = True
|
||||||
|
if specs.get('replication_type') == '<in> sync':
|
||||||
|
rinfo['sync'] = True
|
||||||
|
if specs.get('replication:livevolume') == '<is> True':
|
||||||
|
rinfo['live'] = True
|
||||||
|
if specs.get('replication:activereplay') == '<is> True':
|
||||||
|
rinfo['active'] = True
|
||||||
|
|
||||||
|
# Some quick checks.
|
||||||
|
if rinfo['enabled']:
|
||||||
|
replication_target_count = len(self.backends)
|
||||||
|
msg = None
|
||||||
|
if replication_target_count == 0:
|
||||||
|
msg = _(
|
||||||
|
'Replication setup failure: replication has been '
|
||||||
|
'enabled but no replication target has been specified '
|
||||||
|
'for this backend.')
|
||||||
|
if rinfo['live'] and replication_target_count != 1:
|
||||||
|
msg = _('Replication setup failure: replication:livevolume'
|
||||||
|
' has been enabled but more than one replication '
|
||||||
|
'target has been specified for this backend.')
|
||||||
|
if msg:
|
||||||
|
LOG.debug(msg)
|
||||||
|
raise exception.ReplicationError(message=msg)
|
||||||
|
# Got this far. Life is good. Return our data.
|
||||||
|
return rinfo
|
||||||
|
|
||||||
|
def _is_live_vol(self, api, volume):
|
||||||
|
sclivevolume = None
|
||||||
|
rspecs = self._get_replication_specs(volume)
|
||||||
|
if rspecs['enabled'] and rspecs['live']:
|
||||||
|
# Find our volume and server.
|
||||||
|
sclivevolume = api.get_live_volume(volume['provider_id'])
|
||||||
|
return sclivevolume
|
||||||
|
|
||||||
def _create_replications(self, api, volume, scvolume):
|
def _create_replications(self, api, volume, scvolume):
|
||||||
"""Creates any appropriate replications for a given volume.
|
"""Creates any appropriate replications for a given volume.
|
||||||
@ -188,23 +219,32 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
# for now we assume we have an array named backends.
|
# for now we assume we have an array named backends.
|
||||||
replication_driver_data = None
|
replication_driver_data = None
|
||||||
# Replicate if we are supposed to.
|
# Replicate if we are supposed to.
|
||||||
do_repl, sync = self._do_repl(api, volume)
|
rspecs = self._get_replication_specs(volume)
|
||||||
if do_repl:
|
if rspecs['enabled']:
|
||||||
for backend in self.backends:
|
for backend in self.backends:
|
||||||
# Check if we are to replicate the active replay or not.
|
targetdeviceid = backend['target_device_id']
|
||||||
specs = self._get_volume_extra_specs(volume)
|
primaryqos = backend.get('qosnode', 'cinderqos')
|
||||||
replact = specs.get('replication:activereplay') == '<is> True'
|
secondaryqos = backend.get('remoteqos', 'cinderqos')
|
||||||
if not api.create_replication(scvolume,
|
diskfolder = backend.get('diskfolder', None)
|
||||||
backend['target_device_id'],
|
obj = None
|
||||||
backend.get('qosnode',
|
if rspecs['live']:
|
||||||
'cinderqos'),
|
# We are rolling with a live volume.
|
||||||
sync,
|
obj = api.create_live_volume(scvolume, targetdeviceid,
|
||||||
backend.get('diskfolder', None),
|
rspecs['active'],
|
||||||
replact):
|
rspecs['sync'],
|
||||||
|
primaryqos, secondaryqos)
|
||||||
|
else:
|
||||||
|
# Else a regular replication.
|
||||||
|
obj = api.create_replication(scvolume, targetdeviceid,
|
||||||
|
primaryqos, rspecs['sync'],
|
||||||
|
diskfolder, rspecs['active'])
|
||||||
|
# This is either a ScReplication object or a ScLiveVolume
|
||||||
|
# object. So long as it isn't None we are fine.
|
||||||
|
if not obj:
|
||||||
# Create replication will have printed a better error.
|
# Create replication will have printed a better error.
|
||||||
msg = _('Replication %(name)s to %(ssn)s failed.') % {
|
msg = _('Replication %(name)s to %(ssn)s failed.') % {
|
||||||
'name': volume['id'],
|
'name': volume['id'],
|
||||||
'ssn': backend['target_device_id']}
|
'ssn': targetdeviceid}
|
||||||
raise exception.VolumeBackendAPIException(data=msg)
|
raise exception.VolumeBackendAPIException(data=msg)
|
||||||
if not replication_driver_data:
|
if not replication_driver_data:
|
||||||
replication_driver_data = backend['target_device_id']
|
replication_driver_data = backend['target_device_id']
|
||||||
@ -295,6 +335,40 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
ssnstrings.append(ssnstring)
|
ssnstrings.append(ssnstring)
|
||||||
return ssnstrings
|
return ssnstrings
|
||||||
|
|
||||||
|
def _delete_live_volume(self, api, volume):
|
||||||
|
"""Delete live volume associated with volume.
|
||||||
|
|
||||||
|
:param api:Dell REST API object.
|
||||||
|
:param volume: Cinder Volume object
|
||||||
|
:return: True if we actually deleted something. False for everything
|
||||||
|
else.
|
||||||
|
"""
|
||||||
|
# Live Volume was added after provider_id support. So just assume it is
|
||||||
|
# there.
|
||||||
|
replication_driver_data = volume.get('replication_driver_data')
|
||||||
|
# Do we have any replication driver data?
|
||||||
|
if replication_driver_data:
|
||||||
|
# Valid replication data?
|
||||||
|
ssnstrings = self._split_driver_data(replication_driver_data)
|
||||||
|
if ssnstrings:
|
||||||
|
ssn = int(ssnstrings[0])
|
||||||
|
sclivevolume = api.get_live_volume(volume.get('provider_id'))
|
||||||
|
# Have we found the live volume?
|
||||||
|
if (sclivevolume and
|
||||||
|
sclivevolume.get('secondaryScSerialNumber') == ssn and
|
||||||
|
api.delete_live_volume(sclivevolume, True)):
|
||||||
|
LOG.info(_LI('%(vname)s\'s replication live volume has '
|
||||||
|
'been deleted from storage Center %(sc)s,'),
|
||||||
|
{'vname': volume.get('id'),
|
||||||
|
'sc': ssn})
|
||||||
|
return True
|
||||||
|
# If we are here either we do not have a live volume, we do not have
|
||||||
|
# one on our configured SC or we were not able to delete it.
|
||||||
|
# Either way, warn and leave.
|
||||||
|
LOG.warning(_LW('Unable to delete %s live volume.'),
|
||||||
|
volume.get('id'))
|
||||||
|
return False
|
||||||
|
|
||||||
def _delete_replications(self, api, volume):
|
def _delete_replications(self, api, volume):
|
||||||
"""Delete replications associated with a given volume.
|
"""Delete replications associated with a given volume.
|
||||||
|
|
||||||
@ -304,26 +378,24 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
|
|
||||||
:param api: Dell REST API object.
|
:param api: Dell REST API object.
|
||||||
:param volume: Cinder Volume object
|
:param volume: Cinder Volume object
|
||||||
:return:
|
:return: None
|
||||||
"""
|
"""
|
||||||
do_repl, sync = self._do_repl(api, volume)
|
replication_driver_data = volume.get('replication_driver_data')
|
||||||
if do_repl:
|
if replication_driver_data:
|
||||||
replication_driver_data = volume.get('replication_driver_data')
|
ssnstrings = self._split_driver_data(replication_driver_data)
|
||||||
if replication_driver_data:
|
volume_name = volume.get('id')
|
||||||
ssnstrings = self._split_driver_data(replication_driver_data)
|
provider_id = volume.get('provider_id')
|
||||||
volume_name = volume.get('id')
|
scvol = api.find_volume(volume_name, provider_id)
|
||||||
provider_id = volume.get('provider_id')
|
# This is just a string of ssns separated by commas.
|
||||||
scvol = api.find_volume(volume_name, provider_id)
|
# Trundle through these and delete them all.
|
||||||
# This is just a string of ssns separated by commas.
|
for ssnstring in ssnstrings:
|
||||||
# Trundle through these and delete them all.
|
ssn = int(ssnstring)
|
||||||
for ssnstring in ssnstrings:
|
# Are we a replication or a live volume?
|
||||||
ssn = int(ssnstring)
|
if not api.delete_replication(scvol, ssn):
|
||||||
if not api.delete_replication(scvol, ssn):
|
LOG.warning(_LW('Unable to delete replication of Volume '
|
||||||
LOG.warning(_LW('Unable to delete replication of '
|
'%(vname)s to Storage Center %(sc)s.'),
|
||||||
'Volume %(vname)s to Storage Center '
|
{'vname': volume_name,
|
||||||
'%(sc)s.'),
|
'sc': ssnstring})
|
||||||
{'vname': volume_name,
|
|
||||||
'sc': ssnstring})
|
|
||||||
# If none of that worked or there was nothing to do doesn't matter.
|
# If none of that worked or there was nothing to do doesn't matter.
|
||||||
# Just move on.
|
# Just move on.
|
||||||
|
|
||||||
@ -335,7 +407,12 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
LOG.debug('Deleting volume %s', volume_name)
|
LOG.debug('Deleting volume %s', volume_name)
|
||||||
with self._client.open_connection() as api:
|
with self._client.open_connection() as api:
|
||||||
try:
|
try:
|
||||||
self._delete_replications(api, volume)
|
rspecs = self._get_replication_specs(volume)
|
||||||
|
if rspecs['enabled']:
|
||||||
|
if rspecs['live']:
|
||||||
|
self._delete_live_volume(api, volume)
|
||||||
|
else:
|
||||||
|
self._delete_replications(api, volume)
|
||||||
deleted = api.delete_volume(volume_name, provider_id)
|
deleted = api.delete_volume(volume_name, provider_id)
|
||||||
except Exception:
|
except Exception:
|
||||||
with excutils.save_and_reraise_exception():
|
with excutils.save_and_reraise_exception():
|
||||||
@ -1241,6 +1318,78 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
'updates': model_update})
|
'updates': model_update})
|
||||||
return volume_updates
|
return volume_updates
|
||||||
|
|
||||||
|
def _failback_replication(self, api, volume, qosnode):
|
||||||
|
"""Sets up the replication failback.
|
||||||
|
|
||||||
|
:param api: Dell SC API.
|
||||||
|
:param volume: Cinder Volume
|
||||||
|
:param qosnode: Dell QOS node object.
|
||||||
|
:return: replitem dict.
|
||||||
|
"""
|
||||||
|
LOG.info(_LI('failback_volumes: replicated volume'))
|
||||||
|
# Get our current volume.
|
||||||
|
cvol = api.find_volume(volume['id'], volume['provider_id'])
|
||||||
|
# Original volume on the primary.
|
||||||
|
ovol = api.find_repl_volume(volume['id'], api.primaryssn,
|
||||||
|
None, True, False)
|
||||||
|
# Delete our current mappings.
|
||||||
|
api.remove_mappings(cvol)
|
||||||
|
# If there is a replication to delete do so.
|
||||||
|
api.delete_replication(ovol, api.ssn, False)
|
||||||
|
# Replicate to a common replay.
|
||||||
|
screpl = api.replicate_to_common(cvol, ovol, 'tempqos')
|
||||||
|
# We made it this far. Update our status.
|
||||||
|
screplid = None
|
||||||
|
status = ''
|
||||||
|
if screpl:
|
||||||
|
screplid = screpl['instanceId']
|
||||||
|
nvolid = screpl['destinationVolume']['instanceId']
|
||||||
|
status = 'inprogress'
|
||||||
|
else:
|
||||||
|
LOG.error(_LE('Unable to restore %s'), volume['id'])
|
||||||
|
screplid = None
|
||||||
|
nvolid = None
|
||||||
|
status = 'error'
|
||||||
|
|
||||||
|
# Save some information for the next step.
|
||||||
|
# nvol is the new volume created by replicate_to_common.
|
||||||
|
# We also grab our extra specs here.
|
||||||
|
replitem = {
|
||||||
|
'volume': volume,
|
||||||
|
'specs': self._parse_extraspecs(volume),
|
||||||
|
'qosnode': qosnode,
|
||||||
|
'screpl': screplid,
|
||||||
|
'cvol': cvol['instanceId'],
|
||||||
|
'ovol': ovol['instanceId'],
|
||||||
|
'nvol': nvolid,
|
||||||
|
'rdd': six.text_type(api.ssn),
|
||||||
|
'status': status}
|
||||||
|
|
||||||
|
return replitem
|
||||||
|
|
||||||
|
def _failback_live_volume(self, api, id, provider_id):
|
||||||
|
"""failback the live volume to its original
|
||||||
|
|
||||||
|
:param api: Dell SC API
|
||||||
|
:param id: Volume ID
|
||||||
|
:param provider_id: Dell Instance ID
|
||||||
|
:return: model_update dict
|
||||||
|
"""
|
||||||
|
model_update = {}
|
||||||
|
sclivevolume = api.get_live_volume(provider_id)
|
||||||
|
if sclivevolume and api.swap_roles_live_volume(sclivevolume):
|
||||||
|
LOG.info(_LI('Success swapping sclivevolume roles %s'), id)
|
||||||
|
model_update = {
|
||||||
|
'status': 'available',
|
||||||
|
'replication_status': 'enabled',
|
||||||
|
'provider_id':
|
||||||
|
sclivevolume['secondaryVolume']['instanceId']}
|
||||||
|
else:
|
||||||
|
LOG.info(_LI('Failure swapping roles %s'), id)
|
||||||
|
model_update = {'status': 'error'}
|
||||||
|
|
||||||
|
return model_update
|
||||||
|
|
||||||
def failback_volumes(self, volumes):
|
def failback_volumes(self, volumes):
|
||||||
"""This is a generic volume failback.
|
"""This is a generic volume failback.
|
||||||
|
|
||||||
@ -1258,54 +1407,32 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
|
|
||||||
volume_updates = []
|
volume_updates = []
|
||||||
replitems = []
|
replitems = []
|
||||||
screplid = None
|
|
||||||
status = ''
|
|
||||||
# Trundle through the volumes. Update non replicated to alive again
|
# Trundle through the volumes. Update non replicated to alive again
|
||||||
# and reverse the replications for the remaining volumes.
|
# and reverse the replications for the remaining volumes.
|
||||||
for volume in volumes:
|
for volume in volumes:
|
||||||
LOG.info(_LI('failback_volumes: starting volume: %s'), volume)
|
LOG.info(_LI('failback_volumes: starting volume: %s'), volume)
|
||||||
model_update = {}
|
model_update = {}
|
||||||
if volume.get('replication_driver_data'):
|
if volume.get('replication_driver_data'):
|
||||||
LOG.info(_LI('failback_volumes: replicated volume'))
|
rspecs = self._get_replication_specs(volume)
|
||||||
# Get our current volume.
|
if rspecs['live']:
|
||||||
cvol = api.find_volume(volume['id'], volume['provider_id'])
|
model_update = self._failback_live_volume(
|
||||||
# Original volume on the primary.
|
api, volume['id'], volume['provider_id'])
|
||||||
ovol = api.find_repl_volume(volume['id'], api.primaryssn,
|
|
||||||
None, True, False)
|
|
||||||
# Delete our current mappings.
|
|
||||||
api.remove_mappings(cvol)
|
|
||||||
# If there is a replication to delete do so.
|
|
||||||
api.delete_replication(ovol, api.ssn, False)
|
|
||||||
# Replicate to a common replay.
|
|
||||||
screpl = api.replicate_to_common(cvol, ovol, 'tempqos')
|
|
||||||
# We made it this far. Update our status.
|
|
||||||
if screpl:
|
|
||||||
screplid = screpl['instanceId']
|
|
||||||
nvolid = screpl['destinationVolume']['instanceId']
|
|
||||||
status = 'inprogress'
|
|
||||||
else:
|
else:
|
||||||
LOG.error(_LE('Unable to restore %s'), volume['id'])
|
replitem = self._failback_replication(api, volume,
|
||||||
screplid = None
|
qosnode)
|
||||||
nvolid = None
|
|
||||||
status = 'error'
|
|
||||||
|
|
||||||
# Save some information for the next step.
|
# Save some information for the next step.
|
||||||
# nvol is the new volume created by replicate_to_common.
|
# nvol is the new volume created by
|
||||||
# We also grab our extra specs here.
|
# replicate_to_common. We also grab our
|
||||||
replitems.append(
|
# extra specs here.
|
||||||
{'volume': volume,
|
replitems.append(replitem)
|
||||||
'specs': self._parse_extraspecs(volume),
|
|
||||||
'qosnode': qosnode,
|
|
||||||
'screpl': screplid,
|
|
||||||
'cvol': cvol['instanceId'],
|
|
||||||
'ovol': ovol['instanceId'],
|
|
||||||
'nvol': nvolid,
|
|
||||||
'rdd': six.text_type(api.ssn),
|
|
||||||
'status': status})
|
|
||||||
else:
|
else:
|
||||||
# Not replicated. Just set it to available.
|
# Not replicated. Just set it to available.
|
||||||
model_update = {'status': 'available'}
|
model_update = {'status': 'available'}
|
||||||
# Either we are failed over or our status is now error.
|
|
||||||
|
# Save our update
|
||||||
|
if model_update:
|
||||||
volume_updates.append({'volume_id': volume['id'],
|
volume_updates.append({'volume_id': volume['id'],
|
||||||
'updates': model_update})
|
'updates': model_update})
|
||||||
|
|
||||||
@ -1324,6 +1451,33 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
self._update_backend(None)
|
self._update_backend(None)
|
||||||
return volume_updates
|
return volume_updates
|
||||||
|
|
||||||
|
def _failover_replication(self, api, id, provider_id, destssn):
|
||||||
|
rvol = api.break_replication(id, provider_id, destssn)
|
||||||
|
model_update = {}
|
||||||
|
if rvol:
|
||||||
|
LOG.info(_LI('Success failing over volume %s'), id)
|
||||||
|
model_update = {'replication_status': 'failed-over',
|
||||||
|
'provider_id': rvol['instanceId']}
|
||||||
|
else:
|
||||||
|
LOG.info(_LI('Failed failing over volume %s'), id)
|
||||||
|
model_update = {'status': 'error'}
|
||||||
|
|
||||||
|
return model_update
|
||||||
|
|
||||||
|
def _failover_live_volume(self, api, id, provider_id):
|
||||||
|
model_update = {}
|
||||||
|
sclivevolume = api.get_live_volume(provider_id)
|
||||||
|
if sclivevolume and api.swap_roles_live_volume(sclivevolume):
|
||||||
|
LOG.info(_LI('Success swapping sclivevolume roles %s'), id)
|
||||||
|
model_update = {'replication_status': 'failed-over',
|
||||||
|
'provider_id':
|
||||||
|
sclivevolume['secondaryVolume']['instanceId']}
|
||||||
|
else:
|
||||||
|
LOG.info(_LI('Failure swapping roles %s'), id)
|
||||||
|
model_update = {'status': 'error'}
|
||||||
|
|
||||||
|
return model_update
|
||||||
|
|
||||||
def failover_host(self, context, volumes, secondary_id=None):
|
def failover_host(self, context, volumes, secondary_id=None):
|
||||||
"""Failover to secondary.
|
"""Failover to secondary.
|
||||||
|
|
||||||
@ -1341,7 +1495,6 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
'replication_status': 'failed-over',
|
'replication_status': 'failed-over',
|
||||||
'replication_extended_status': 'whatever',...}},]
|
'replication_extended_status': 'whatever',...}},]
|
||||||
"""
|
"""
|
||||||
|
|
||||||
LOG.debug('failover-host')
|
LOG.debug('failover-host')
|
||||||
LOG.debug(self.failed_over)
|
LOG.debug(self.failed_over)
|
||||||
LOG.debug(self.active_backend_id)
|
LOG.debug(self.active_backend_id)
|
||||||
@ -1366,21 +1519,15 @@ class DellCommonDriver(driver.ConsistencyGroupVD, driver.ManageableVD,
|
|||||||
for volume in volumes:
|
for volume in volumes:
|
||||||
model_update = {}
|
model_update = {}
|
||||||
if volume.get('replication_driver_data'):
|
if volume.get('replication_driver_data'):
|
||||||
rvol = api.break_replication(
|
rspecs = self._get_replication_specs(volume)
|
||||||
volume['id'], volume.get('provider_id'),
|
if rspecs['live']:
|
||||||
destssn)
|
model_update = self._failover_live_volume(
|
||||||
if rvol:
|
api, volume['id'],
|
||||||
LOG.info(_LI('Success failing over volume %s'),
|
volume.get('provider_id'))
|
||||||
volume['id'])
|
|
||||||
else:
|
else:
|
||||||
LOG.info(_LI('Failed failing over volume %s'),
|
model_update = self._failover_replication(
|
||||||
volume['id'])
|
api, volume['id'],
|
||||||
|
volume.get('provider_id'), destssn)
|
||||||
# We should note that we are now failed over
|
|
||||||
# and that we have a new instanceId.
|
|
||||||
model_update = {
|
|
||||||
'replication_status': 'failed-over',
|
|
||||||
'provider_id': rvol['instanceId']}
|
|
||||||
else:
|
else:
|
||||||
# Not a replicated volume. Try to unmap it.
|
# Not a replicated volume. Try to unmap it.
|
||||||
scvolume = api.find_volume(
|
scvolume = api.find_volume(
|
||||||
|
@ -18,7 +18,7 @@ from oslo_log import log as logging
|
|||||||
from oslo_utils import excutils
|
from oslo_utils import excutils
|
||||||
|
|
||||||
from cinder import exception
|
from cinder import exception
|
||||||
from cinder.i18n import _, _LE
|
from cinder.i18n import _, _LE, _LW
|
||||||
from cinder import interface
|
from cinder import interface
|
||||||
from cinder.volume import driver
|
from cinder.volume import driver
|
||||||
from cinder.volume.drivers.dell import dell_storagecenter_common
|
from cinder.volume.drivers.dell import dell_storagecenter_common
|
||||||
@ -53,11 +53,12 @@ class DellStorageCenterFCDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
2.4.1 - Updated Replication support to V2.1.
|
2.4.1 - Updated Replication support to V2.1.
|
||||||
2.5.0 - ManageableSnapshotsVD implemented.
|
2.5.0 - ManageableSnapshotsVD implemented.
|
||||||
3.0.0 - ProviderID utilized.
|
3.0.0 - ProviderID utilized.
|
||||||
3.1.0 - Failback Supported.
|
3.1.0 - Failback supported.
|
||||||
|
3.2.0 - Live Volume support.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
VERSION = '3.1.0'
|
VERSION = '3.2.0'
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super(DellStorageCenterFCDriver, self).__init__(*args, **kwargs)
|
super(DellStorageCenterFCDriver, self).__init__(*args, **kwargs)
|
||||||
@ -84,17 +85,13 @@ class DellStorageCenterFCDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
LOG.debug('Initialize connection: %s', volume_name)
|
LOG.debug('Initialize connection: %s', volume_name)
|
||||||
with self._client.open_connection() as api:
|
with self._client.open_connection() as api:
|
||||||
try:
|
try:
|
||||||
# Find our server.
|
|
||||||
scserver = None
|
|
||||||
wwpns = connector.get('wwpns')
|
wwpns = connector.get('wwpns')
|
||||||
for wwn in wwpns:
|
# Find our server.
|
||||||
scserver = api.find_server(wwn)
|
scserver = self._find_server(api, wwpns)
|
||||||
if scserver is not None:
|
|
||||||
break
|
|
||||||
|
|
||||||
# No? Create it.
|
# No? Create it.
|
||||||
if scserver is None:
|
if scserver is None:
|
||||||
scserver = api.create_server_multiple_hbas(wwpns)
|
scserver = api.create_server(wwpns)
|
||||||
# Find the volume on the storage center.
|
# Find the volume on the storage center.
|
||||||
scvolume = api.find_volume(volume_name, provider_id)
|
scvolume = api.find_volume(volume_name, provider_id)
|
||||||
if scserver is not None and scvolume is not None:
|
if scserver is not None and scvolume is not None:
|
||||||
@ -105,6 +102,16 @@ class DellStorageCenterFCDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
scvolume = api.get_volume(scvolume['instanceId'])
|
scvolume = api.get_volume(scvolume['instanceId'])
|
||||||
lun, targets, init_targ_map = api.find_wwns(scvolume,
|
lun, targets, init_targ_map = api.find_wwns(scvolume,
|
||||||
scserver)
|
scserver)
|
||||||
|
sclivevolume = self._is_live_vol(api, volume)
|
||||||
|
if sclivevolume:
|
||||||
|
# Now map our secondary.
|
||||||
|
lvlun, lvtargets, lvinit_targ_map = (
|
||||||
|
self.initialize_secondary(api, sclivevolume,
|
||||||
|
wwpns))
|
||||||
|
# Unmapped. Add info to our list.
|
||||||
|
targets += lvtargets
|
||||||
|
init_targ_map.update(lvinit_targ_map)
|
||||||
|
|
||||||
if lun is not None and len(targets) > 0:
|
if lun is not None and len(targets) > 0:
|
||||||
data = {'driver_volume_type': 'fibre_channel',
|
data = {'driver_volume_type': 'fibre_channel',
|
||||||
'data': {'target_lun': lun,
|
'data': {'target_lun': lun,
|
||||||
@ -124,6 +131,42 @@ class DellStorageCenterFCDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
# We get here because our mapping is none so blow up.
|
# We get here because our mapping is none so blow up.
|
||||||
raise exception.VolumeBackendAPIException(_('Unable to map volume.'))
|
raise exception.VolumeBackendAPIException(_('Unable to map volume.'))
|
||||||
|
|
||||||
|
def _find_server(self, api, wwns, ssn=-1):
|
||||||
|
for wwn in wwns:
|
||||||
|
scserver = api.find_server(wwn, ssn)
|
||||||
|
if scserver is not None:
|
||||||
|
return scserver
|
||||||
|
return None
|
||||||
|
|
||||||
|
def initialize_secondary(self, api, sclivevolume, wwns):
|
||||||
|
"""Initialize the secondary connection of a live volume pair.
|
||||||
|
|
||||||
|
:param api: Dell SC api object.
|
||||||
|
:param sclivevolume: Dell SC live volume object.
|
||||||
|
:param wwns: Cinder list of wwns from the connector.
|
||||||
|
:return: lun, targets and initiator target map.
|
||||||
|
"""
|
||||||
|
# Find our server.
|
||||||
|
secondary = self._find_server(
|
||||||
|
api, wwns, sclivevolume['secondaryScSerialNumber'])
|
||||||
|
|
||||||
|
# No? Create it.
|
||||||
|
if secondary is None:
|
||||||
|
secondary = api.create_server(
|
||||||
|
wwns, sclivevolume['secondaryScSerialNumber'])
|
||||||
|
if secondary:
|
||||||
|
if api.map_secondary_volume(sclivevolume, secondary):
|
||||||
|
# Get mappings.
|
||||||
|
secondaryvol = api.get_volume(
|
||||||
|
sclivevolume['secondaryVolume']['instanceId'])
|
||||||
|
if secondaryvol:
|
||||||
|
return api.find_wwns(secondaryvol, secondary)
|
||||||
|
LOG.warning(_LW('Unable to map live volume secondary volume'
|
||||||
|
' %(vol)s to secondary server wwns: %(wwns)r'),
|
||||||
|
{'vol': sclivevolume['secondaryVolume']['instanceName'],
|
||||||
|
'wwns': wwns})
|
||||||
|
return None, [], {}
|
||||||
|
|
||||||
@fczm_utils.RemoveFCZone
|
@fczm_utils.RemoveFCZone
|
||||||
def terminate_connection(self, volume, connector, force=False, **kwargs):
|
def terminate_connection(self, volume, connector, force=False, **kwargs):
|
||||||
# Get our volume name
|
# Get our volume name
|
||||||
@ -132,17 +175,23 @@ class DellStorageCenterFCDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
LOG.debug('Terminate connection: %s', volume_name)
|
LOG.debug('Terminate connection: %s', volume_name)
|
||||||
with self._client.open_connection() as api:
|
with self._client.open_connection() as api:
|
||||||
try:
|
try:
|
||||||
scserver = None
|
|
||||||
wwpns = connector.get('wwpns')
|
wwpns = connector.get('wwpns')
|
||||||
for wwn in wwpns:
|
scserver = self._find_server(api, wwpns)
|
||||||
scserver = api.find_server(wwn)
|
|
||||||
if scserver is not None:
|
|
||||||
break
|
|
||||||
|
|
||||||
# Find the volume on the storage center.
|
# Find the volume on the storage center.
|
||||||
scvolume = api.find_volume(volume_name, provider_id)
|
scvolume = api.find_volume(volume_name, provider_id)
|
||||||
# Get our target map so we can return it to free up a zone.
|
# Get our target map so we can return it to free up a zone.
|
||||||
lun, targets, init_targ_map = api.find_wwns(scvolume, scserver)
|
lun, targets, init_targ_map = api.find_wwns(scvolume, scserver)
|
||||||
|
# Unmap from our secondary first.
|
||||||
|
sclivevolume = self._is_live_vol(api, volume)
|
||||||
|
if sclivevolume:
|
||||||
|
lvlun, lvtargets, lvinit_targ_map = (
|
||||||
|
self.terminate_secondary(api, sclivevolume, wwpns))
|
||||||
|
# Add to our return.
|
||||||
|
if lvlun:
|
||||||
|
targets += lvtargets
|
||||||
|
init_targ_map.update(lvinit_targ_map)
|
||||||
|
|
||||||
# If we have a server and a volume lets unmap them.
|
# If we have a server and a volume lets unmap them.
|
||||||
if (scserver is not None and
|
if (scserver is not None and
|
||||||
scvolume is not None and
|
scvolume is not None and
|
||||||
@ -168,3 +217,24 @@ class DellStorageCenterFCDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
LOG.error(_LE('Failed to terminate connection'))
|
LOG.error(_LE('Failed to terminate connection'))
|
||||||
raise exception.VolumeBackendAPIException(
|
raise exception.VolumeBackendAPIException(
|
||||||
_('Terminate connection unable to connect to backend.'))
|
_('Terminate connection unable to connect to backend.'))
|
||||||
|
|
||||||
|
def terminate_secondary(self, api, sclivevolume, wwns):
|
||||||
|
# Find our server.
|
||||||
|
secondary = self._find_server(
|
||||||
|
api, wwns, sclivevolume['secondaryScSerialNumber'])
|
||||||
|
secondaryvol = api.get_volume(
|
||||||
|
sclivevolume['secondaryVolume']['instanceId'])
|
||||||
|
if secondary and secondaryvol:
|
||||||
|
# Get our map.
|
||||||
|
lun, targets, init_targ_map = api.find_wwns(secondaryvol,
|
||||||
|
secondary)
|
||||||
|
# If we have a server and a volume lets unmap them.
|
||||||
|
ret = api.unmap_volume(secondaryvol, secondary)
|
||||||
|
LOG.debug('terminate_secondary: secondary volume %(name)s unmap '
|
||||||
|
'to secondary server %(server)s result: %(result)r',
|
||||||
|
{'name': secondaryvol['name'],
|
||||||
|
'server': secondary['name'],
|
||||||
|
'result': ret})
|
||||||
|
# return info for
|
||||||
|
return lun, targets, init_targ_map
|
||||||
|
return None, [], {}
|
||||||
|
@ -18,7 +18,7 @@ from oslo_log import log as logging
|
|||||||
from oslo_utils import excutils
|
from oslo_utils import excutils
|
||||||
|
|
||||||
from cinder import exception
|
from cinder import exception
|
||||||
from cinder.i18n import _, _LE, _LI
|
from cinder.i18n import _, _LE, _LI, _LW
|
||||||
from cinder import interface
|
from cinder import interface
|
||||||
from cinder.volume import driver
|
from cinder.volume import driver
|
||||||
from cinder.volume.drivers.dell import dell_storagecenter_common
|
from cinder.volume.drivers.dell import dell_storagecenter_common
|
||||||
@ -53,10 +53,11 @@ class DellStorageCenterISCSIDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
2.5.0 - ManageableSnapshotsVD implemented.
|
2.5.0 - ManageableSnapshotsVD implemented.
|
||||||
3.0.0 - ProviderID utilized.
|
3.0.0 - ProviderID utilized.
|
||||||
3.1.0 - Failback Supported.
|
3.1.0 - Failback Supported.
|
||||||
|
3.2.0 - Live Volume support.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
VERSION = '3.1.0'
|
VERSION = '3.2.0'
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super(DellStorageCenterISCSIDriver, self).__init__(*args, **kwargs)
|
super(DellStorageCenterISCSIDriver, self).__init__(*args, **kwargs)
|
||||||
@ -83,39 +84,32 @@ class DellStorageCenterISCSIDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
provider_id = volume.get('provider_id')
|
provider_id = volume.get('provider_id')
|
||||||
initiator_name = connector.get('initiator')
|
initiator_name = connector.get('initiator')
|
||||||
multipath = connector.get('multipath', False)
|
multipath = connector.get('multipath', False)
|
||||||
LOG.info(_LI('initialize_ connection: %(vol)s:%(initiator)s'),
|
LOG.info(_LI('initialize_ connection: %(vol)s:%(pid)s:'
|
||||||
|
'%(intr)s. Multipath is %(mp)r'),
|
||||||
{'vol': volume_name,
|
{'vol': volume_name,
|
||||||
'initiator': initiator_name})
|
'pid': provider_id,
|
||||||
|
'intr': initiator_name,
|
||||||
|
'mp': multipath})
|
||||||
|
|
||||||
with self._client.open_connection() as api:
|
with self._client.open_connection() as api:
|
||||||
try:
|
try:
|
||||||
# Find our server.
|
# Find our server.
|
||||||
server = api.find_server(initiator_name)
|
scserver = api.find_server(initiator_name)
|
||||||
# No? Create it.
|
# No? Create it.
|
||||||
if server is None:
|
if scserver is None:
|
||||||
server = api.create_server(initiator_name)
|
scserver = api.create_server([initiator_name])
|
||||||
# Find the volume on the storage center.
|
# Find the volume on the storage center.
|
||||||
scvolume = api.find_volume(volume_name, provider_id)
|
scvolume = api.find_volume(volume_name, provider_id)
|
||||||
|
|
||||||
# if we have a server and a volume lets bring them together.
|
# if we have a server and a volume lets bring them together.
|
||||||
if server is not None and scvolume is not None:
|
if scserver is not None and scvolume is not None:
|
||||||
mapping = api.map_volume(scvolume,
|
mapping = api.map_volume(scvolume, scserver)
|
||||||
server)
|
|
||||||
if mapping is not None:
|
if mapping is not None:
|
||||||
# Since we just mapped our volume we had best update
|
# Since we just mapped our volume we had best update
|
||||||
# our sc volume object.
|
# our sc volume object.
|
||||||
scvolume = api.get_volume(provider_id)
|
scvolume = api.get_volume(provider_id)
|
||||||
# Our return.
|
# Our return.
|
||||||
iscsiprops = {}
|
iscsiprops = {}
|
||||||
ip = None
|
|
||||||
port = None
|
|
||||||
if not multipath:
|
|
||||||
# We want to make sure we point to the specified
|
|
||||||
# ip address for our target_portal return. This
|
|
||||||
# isn't an issue with multipath since it should
|
|
||||||
# try all the alternate portal.
|
|
||||||
ip = self.configuration.iscsi_ip_address
|
|
||||||
port = self.configuration.iscsi_port
|
|
||||||
|
|
||||||
# Three cases that should all be satisfied with the
|
# Three cases that should all be satisfied with the
|
||||||
# same return of Target_Portal and Target_Portals.
|
# same return of Target_Portal and Target_Portals.
|
||||||
@ -128,10 +122,22 @@ class DellStorageCenterISCSIDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
# 3. OS brick is calling us in single path mode so
|
# 3. OS brick is calling us in single path mode so
|
||||||
# we want to return Target_Portal and
|
# we want to return Target_Portal and
|
||||||
# Target_Portals as alternates.
|
# Target_Portals as alternates.
|
||||||
iscsiprops = (api.find_iscsi_properties(scvolume,
|
iscsiprops = api.find_iscsi_properties(scvolume)
|
||||||
ip,
|
|
||||||
port))
|
|
||||||
|
|
||||||
|
# If this is a live volume we need to map up our
|
||||||
|
# secondary volume.
|
||||||
|
sclivevolume = self._is_live_vol(api, volume)
|
||||||
|
if sclivevolume:
|
||||||
|
secondaryprops = self.initialize_secondary(
|
||||||
|
api, sclivevolume, initiator_name)
|
||||||
|
# Combine with iscsiprops
|
||||||
|
iscsiprops['target_iqns'] += (
|
||||||
|
secondaryprops['target_iqns'])
|
||||||
|
iscsiprops['target_portals'] += (
|
||||||
|
secondaryprops['target_portals'])
|
||||||
|
iscsiprops['target_luns'] += (
|
||||||
|
secondaryprops['target_luns'])
|
||||||
|
# TODO(tswanson): Get non multipath info from primary.
|
||||||
# Return our iscsi properties.
|
# Return our iscsi properties.
|
||||||
iscsiprops['discard'] = True
|
iscsiprops['discard'] = True
|
||||||
return {'driver_volume_type': 'iscsi',
|
return {'driver_volume_type': 'iscsi',
|
||||||
@ -151,6 +157,44 @@ class DellStorageCenterISCSIDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
raise exception.VolumeBackendAPIException(
|
raise exception.VolumeBackendAPIException(
|
||||||
_('Unable to map volume'))
|
_('Unable to map volume'))
|
||||||
|
|
||||||
|
def initialize_secondary(self, api, sclivevolume, initiatorname):
|
||||||
|
"""Initialize the secondary connection of a live volume pair.
|
||||||
|
|
||||||
|
:param api: Dell SC api.
|
||||||
|
:param sclivevolume: Dell SC live volume object.
|
||||||
|
:param initiatorname: Cinder iscsi initiator from the connector.
|
||||||
|
:return: ISCSI properties.
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Find our server.
|
||||||
|
secondary = api.find_server(initiatorname,
|
||||||
|
sclivevolume['secondaryScSerialNumber'])
|
||||||
|
# No? Create it.
|
||||||
|
if secondary is None:
|
||||||
|
secondary = api.create_server(
|
||||||
|
[initiatorname], sclivevolume['secondaryScSerialNumber'])
|
||||||
|
if secondary:
|
||||||
|
if api.map_secondary_volume(sclivevolume, secondary):
|
||||||
|
# Get our volume and get our properties.
|
||||||
|
secondaryvol = api.get_volume(
|
||||||
|
sclivevolume['secondaryVolume']['instanceId'])
|
||||||
|
if secondaryvol:
|
||||||
|
return api.find_iscsi_properties(secondaryvol)
|
||||||
|
# Dummy return on failure.
|
||||||
|
data = {'target_discovered': False,
|
||||||
|
'target_iqn': None,
|
||||||
|
'target_iqns': [],
|
||||||
|
'target_portal': None,
|
||||||
|
'target_portals': [],
|
||||||
|
'target_lun': None,
|
||||||
|
'target_luns': [],
|
||||||
|
}
|
||||||
|
LOG.warning(_LW('Unable to map live volume secondary volume'
|
||||||
|
' %(vol)s to secondary server intiator: %(init)r'),
|
||||||
|
{'vol': sclivevolume['secondaryVolume']['instanceName'],
|
||||||
|
'init': initiatorname})
|
||||||
|
return data
|
||||||
|
|
||||||
def terminate_connection(self, volume, connector, force=False, **kwargs):
|
def terminate_connection(self, volume, connector, force=False, **kwargs):
|
||||||
# Grab some initial info.
|
# Grab some initial info.
|
||||||
initiator_name = connector.get('initiator')
|
initiator_name = connector.get('initiator')
|
||||||
@ -165,6 +209,10 @@ class DellStorageCenterISCSIDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
# Find the volume on the storage center.
|
# Find the volume on the storage center.
|
||||||
scvolume = api.find_volume(volume_name, provider_id)
|
scvolume = api.find_volume(volume_name, provider_id)
|
||||||
|
|
||||||
|
sclivevolume = self._is_live_vol(api, volume)
|
||||||
|
if sclivevolume:
|
||||||
|
self.terminate_secondary(api, sclivevolume, initiator_name)
|
||||||
|
|
||||||
# If we have a server and a volume lets pull them apart.
|
# If we have a server and a volume lets pull them apart.
|
||||||
if (scserver is not None and
|
if (scserver is not None and
|
||||||
scvolume is not None and
|
scvolume is not None and
|
||||||
@ -179,3 +227,11 @@ class DellStorageCenterISCSIDriver(dell_storagecenter_common.DellCommonDriver,
|
|||||||
'vol': volume_name})
|
'vol': volume_name})
|
||||||
raise exception.VolumeBackendAPIException(
|
raise exception.VolumeBackendAPIException(
|
||||||
_('Terminate connection failed'))
|
_('Terminate connection failed'))
|
||||||
|
|
||||||
|
def terminate_secondary(self, api, sclivevolume, initiatorname):
|
||||||
|
# Find our server.
|
||||||
|
secondary = api.find_server(initiatorname,
|
||||||
|
sclivevolume['secondaryScSerialNumber'])
|
||||||
|
secondaryvol = api.get_volume(
|
||||||
|
sclivevolume['secondaryVolume']['instanceId'])
|
||||||
|
return api.unmap_volume(secondaryvol, secondary)
|
||||||
|
@ -0,0 +1,4 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- Added support for the use of live volume in place of
|
||||||
|
standard replication in the Dell SC driver.
|
Loading…
Reference in New Issue
Block a user