NetApp ONTAP: Fixed errors on failover-host operation with REST API
Fixed the functions _set_snapmirror_state and break_snapmirror that were causing a failure in the failover-host operation. Change-Id: Ib4ebe3cafb67a16aef4b961ccf30dade99fdec7f Closes-Bug: #2028857
This commit is contained in:
parent
0dc8d75229
commit
64b5543d92
@ -2865,6 +2865,7 @@ SNAPMIRROR_GET_ITER_RESPONSE_REST = {
|
|||||||
"type": "async"
|
"type": "async"
|
||||||
},
|
},
|
||||||
"state": "snapmirrored",
|
"state": "snapmirrored",
|
||||||
|
"transfer": {"state": "success"},
|
||||||
"healthy": True
|
"healthy": True
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
@ -2957,10 +2958,11 @@ REST_GET_SNAPMIRRORS_RESPONSE = [{
|
|||||||
'lag-time': None,
|
'lag-time': None,
|
||||||
'last-transfer-end-timestamp': None,
|
'last-transfer-end-timestamp': None,
|
||||||
'mirror-state': 'snapmirrored',
|
'mirror-state': 'snapmirrored',
|
||||||
'relationship-status': 'snapmirrored',
|
'relationship-status': 'idle',
|
||||||
'source-volume': SM_SOURCE_VOLUME,
|
'source-volume': SM_SOURCE_VOLUME,
|
||||||
'source-vserver': SM_SOURCE_VSERVER,
|
'source-vserver': SM_SOURCE_VSERVER,
|
||||||
'uuid': FAKE_UUID,
|
'uuid': FAKE_UUID,
|
||||||
|
'transferring-state': 'success',
|
||||||
}]
|
}]
|
||||||
|
|
||||||
TRANSFERS_GET_ITER_REST = {
|
TRANSFERS_GET_ITER_REST = {
|
||||||
|
@ -2737,8 +2737,8 @@ class NetAppRestCmodeClientTestCase(test.TestCase):
|
|||||||
'destination.path': (fake_client.SM_DEST_VSERVER +
|
'destination.path': (fake_client.SM_DEST_VSERVER +
|
||||||
':' + fake_client.SM_DEST_VOLUME),
|
':' + fake_client.SM_DEST_VOLUME),
|
||||||
'fields': 'state,source.svm.name,source.path,destination.svm.name,'
|
'fields': 'state,source.svm.name,source.path,destination.svm.name,'
|
||||||
'destination.path,transfer.end_time,lag_time,healthy,'
|
'destination.path,transfer.state,transfer.end_time,'
|
||||||
'uuid'
|
'lag_time,healthy,uuid'
|
||||||
}
|
}
|
||||||
|
|
||||||
mock_send_request.assert_called_once_with('/snapmirror/relationships',
|
mock_send_request.assert_called_once_with('/snapmirror/relationships',
|
||||||
@ -2762,8 +2762,8 @@ class NetAppRestCmodeClientTestCase(test.TestCase):
|
|||||||
'destination.path': (fake_client.SM_DEST_VSERVER +
|
'destination.path': (fake_client.SM_DEST_VSERVER +
|
||||||
':' + fake_client.SM_DEST_VOLUME),
|
':' + fake_client.SM_DEST_VOLUME),
|
||||||
'fields': 'state,source.svm.name,source.path,destination.svm.name,'
|
'fields': 'state,source.svm.name,source.path,destination.svm.name,'
|
||||||
'destination.path,transfer.end_time,lag_time,healthy,'
|
'destination.path,transfer.state,transfer.end_time,'
|
||||||
'uuid'
|
'lag_time,healthy,uuid'
|
||||||
}
|
}
|
||||||
|
|
||||||
mock_send_request.assert_called_once_with('/snapmirror/relationships',
|
mock_send_request.assert_called_once_with('/snapmirror/relationships',
|
||||||
@ -2789,8 +2789,8 @@ class NetAppRestCmodeClientTestCase(test.TestCase):
|
|||||||
'destination.path': (fake_client.SM_DEST_VSERVER +
|
'destination.path': (fake_client.SM_DEST_VSERVER +
|
||||||
':' + fake_client.SM_DEST_VOLUME),
|
':' + fake_client.SM_DEST_VOLUME),
|
||||||
'fields': 'state,source.svm.name,source.path,destination.svm.name,'
|
'fields': 'state,source.svm.name,source.path,destination.svm.name,'
|
||||||
'destination.path,transfer.end_time,lag_time,healthy,'
|
'destination.path,transfer.state,transfer.end_time,'
|
||||||
'uuid'
|
'lag_time,healthy,uuid'
|
||||||
}
|
}
|
||||||
|
|
||||||
mock_send_request.assert_called_once_with('/snapmirror/relationships',
|
mock_send_request.assert_called_once_with('/snapmirror/relationships',
|
||||||
@ -3233,37 +3233,23 @@ class NetAppRestCmodeClientTestCase(test.TestCase):
|
|||||||
self.assertEqual(expected_job, result)
|
self.assertEqual(expected_job, result)
|
||||||
|
|
||||||
def test_break_snapmirror(self):
|
def test_break_snapmirror(self):
|
||||||
snapmirror_response = copy.deepcopy(
|
fake_snapmirror = fake_client.REST_GET_SNAPMIRRORS_RESPONSE
|
||||||
fake_client.SNAPMIRROR_GET_ITER_RESPONSE_REST)
|
fake_uuid = fake_snapmirror[0]['uuid']
|
||||||
|
fake_body = {'state': 'broken_off'}
|
||||||
|
|
||||||
snapmirror_response['state'] = 'broken_off'
|
self.mock_object(self.client, 'send_request')
|
||||||
response_list = [snapmirror_response]
|
|
||||||
|
|
||||||
self.mock_object(self.client, 'send_request',
|
mock_get_snap = self.mock_object(
|
||||||
side_effect=copy.deepcopy(response_list))
|
self.client, '_get_snapmirrors',
|
||||||
|
mock.Mock(return_value=fake_snapmirror))
|
||||||
expected_job = {
|
|
||||||
'operation-id': None,
|
|
||||||
'status': None,
|
|
||||||
'jobid': fake_client.FAKE_UUID,
|
|
||||||
'error-code': None,
|
|
||||||
'error-message': None,
|
|
||||||
'relationship-uuid': fake_client.FAKE_UUID,
|
|
||||||
}
|
|
||||||
|
|
||||||
mock_set_snapmirror_state = self.mock_object(
|
|
||||||
self.client,
|
|
||||||
'_set_snapmirror_state',
|
|
||||||
return_value=expected_job)
|
|
||||||
|
|
||||||
self.client.break_snapmirror(
|
self.client.break_snapmirror(
|
||||||
fake_client.SM_SOURCE_VSERVER, fake_client.SM_SOURCE_VOLUME,
|
fake_client.SM_SOURCE_VSERVER, fake_client.SM_SOURCE_VOLUME,
|
||||||
fake_client.SM_DEST_VSERVER, fake_client.SM_DEST_VOLUME)
|
fake_client.SM_DEST_VSERVER, fake_client.SM_DEST_VOLUME)
|
||||||
|
|
||||||
mock_set_snapmirror_state.assert_called_once_with(
|
mock_get_snap.assert_called_once()
|
||||||
'broken-off',
|
self.client.send_request.assert_called_once_with(
|
||||||
fake_client.SM_SOURCE_VSERVER, fake_client.SM_SOURCE_VOLUME,
|
f'/snapmirror/relationships/{fake_uuid}', 'patch', body=fake_body)
|
||||||
fake_client.SM_DEST_VSERVER, fake_client.SM_DEST_VOLUME)
|
|
||||||
|
|
||||||
def test_break_snapmirror_not_found(self):
|
def test_break_snapmirror_not_found(self):
|
||||||
self.mock_object(
|
self.mock_object(
|
||||||
@ -3278,13 +3264,29 @@ class NetAppRestCmodeClientTestCase(test.TestCase):
|
|||||||
fake_client.SM_DEST_VSERVER,
|
fake_client.SM_DEST_VSERVER,
|
||||||
fake_client.SM_DEST_VOLUME)
|
fake_client.SM_DEST_VOLUME)
|
||||||
|
|
||||||
def test_break_snapmirror_timeout(self):
|
def test__break_snapmirror_error(self):
|
||||||
# when a timeout happens, an exception is thrown by send_request
|
fake_snapmirror = fake_client.REST_GET_SNAPMIRRORS_RESPONSE
|
||||||
api_error = netapp_api.NaRetryableError()
|
self.mock_object(self.client, '_get_snapmirrors',
|
||||||
|
return_value=fake_snapmirror)
|
||||||
self.mock_object(self.client, 'send_request',
|
self.mock_object(self.client, 'send_request',
|
||||||
side_effect=api_error)
|
side_effect=self._mock_api_error())
|
||||||
|
self.assertRaises(netapp_api.NaApiError,
|
||||||
|
self.client.break_snapmirror,
|
||||||
|
fake_client.SM_SOURCE_VSERVER,
|
||||||
|
fake_client.SM_SOURCE_VOLUME,
|
||||||
|
fake_client.SM_DEST_VSERVER,
|
||||||
|
fake_client.SM_DEST_VOLUME)
|
||||||
|
|
||||||
self.assertRaises(netapp_api.NaRetryableError,
|
def test__break_snapmirror_exception(self):
|
||||||
|
fake_snapmirror = copy.deepcopy(
|
||||||
|
fake_client.REST_GET_SNAPMIRRORS_RESPONSE)
|
||||||
|
fake_snapmirror[0]['transferring-state'] = 'error'
|
||||||
|
|
||||||
|
self.mock_object(
|
||||||
|
self.client, '_get_snapmirrors',
|
||||||
|
mock.Mock(return_value=fake_snapmirror))
|
||||||
|
|
||||||
|
self.assertRaises(netapp_utils.NetAppDriverException,
|
||||||
self.client.break_snapmirror,
|
self.client.break_snapmirror,
|
||||||
fake_client.SM_SOURCE_VSERVER,
|
fake_client.SM_SOURCE_VSERVER,
|
||||||
fake_client.SM_SOURCE_VOLUME,
|
fake_client.SM_SOURCE_VOLUME,
|
||||||
|
@ -1956,7 +1956,7 @@ class RestClient(object):
|
|||||||
destination_vserver=None, destination_volume=None):
|
destination_vserver=None, destination_volume=None):
|
||||||
|
|
||||||
fields = ['state', 'source.svm.name', 'source.path',
|
fields = ['state', 'source.svm.name', 'source.path',
|
||||||
'destination.svm.name', 'destination.path',
|
'destination.svm.name', 'destination.path', 'transfer.state',
|
||||||
'transfer.end_time', 'lag_time', 'healthy', 'uuid']
|
'transfer.end_time', 'lag_time', 'healthy', 'uuid']
|
||||||
|
|
||||||
query = {}
|
query = {}
|
||||||
@ -1976,7 +1976,11 @@ class RestClient(object):
|
|||||||
snapmirrors = []
|
snapmirrors = []
|
||||||
for record in response.get('records', []):
|
for record in response.get('records', []):
|
||||||
snapmirrors.append({
|
snapmirrors.append({
|
||||||
'relationship-status': record.get('state'),
|
'relationship-status': (
|
||||||
|
'idle'
|
||||||
|
if record.get('state') == 'snapmirrored'
|
||||||
|
else record.get('state')),
|
||||||
|
'transferring-state': record.get('transfer', {}).get('state'),
|
||||||
'mirror-state': record['state'],
|
'mirror-state': record['state'],
|
||||||
'source-vserver': record['source']['svm']['name'],
|
'source-vserver': record['source']['svm']['name'],
|
||||||
'source-volume': (record['source']['path'].split(':')[1] if
|
'source-volume': (record['source']['path'].split(':')[1] if
|
||||||
@ -2070,11 +2074,11 @@ class RestClient(object):
|
|||||||
result = self.send_request('/snapmirror/relationships/' + uuid,
|
result = self.send_request('/snapmirror/relationships/' + uuid,
|
||||||
'patch', body=body,
|
'patch', body=body,
|
||||||
wait_on_accepted=wait_result)
|
wait_on_accepted=wait_result)
|
||||||
job = result['job']
|
|
||||||
job_info = {
|
job_info = {
|
||||||
'operation-id': None,
|
'operation-id': None,
|
||||||
'status': None,
|
'status': None,
|
||||||
'jobid': job.get('uuid'),
|
'jobid': result.get('job', {}).get('uuid'),
|
||||||
'error-code': None,
|
'error-code': None,
|
||||||
'error-message': None,
|
'error-message': None,
|
||||||
'relationship-uuid': uuid,
|
'relationship-uuid': uuid,
|
||||||
@ -2247,9 +2251,39 @@ class RestClient(object):
|
|||||||
destination_vserver, destination_volume):
|
destination_vserver, destination_volume):
|
||||||
"""Breaks a data protection SnapMirror relationship."""
|
"""Breaks a data protection SnapMirror relationship."""
|
||||||
|
|
||||||
self._set_snapmirror_state(
|
interval = 2
|
||||||
'broken-off', source_vserver, source_volume,
|
retries = (10 / interval)
|
||||||
destination_vserver, destination_volume)
|
|
||||||
|
@utils.retry(netapp_api.NaRetryableError, interval=interval,
|
||||||
|
retries=retries, backoff_rate=1)
|
||||||
|
def _waiter():
|
||||||
|
snapmirror = self.get_snapmirrors(
|
||||||
|
source_vserver=source_vserver,
|
||||||
|
source_volume=source_volume,
|
||||||
|
destination_vserver=destination_vserver,
|
||||||
|
destination_volume=destination_volume)
|
||||||
|
|
||||||
|
snapmirror_state = None
|
||||||
|
if snapmirror:
|
||||||
|
snapmirror_state = snapmirror[0].get('transferring-state')
|
||||||
|
|
||||||
|
if snapmirror_state == 'success':
|
||||||
|
uuid = snapmirror[0]['uuid']
|
||||||
|
body = {'state': 'broken_off'}
|
||||||
|
self.send_request(f'/snapmirror/relationships/{uuid}', 'patch',
|
||||||
|
body=body)
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
message = 'Waiting for transfer state to be SUCCESS.'
|
||||||
|
code = ''
|
||||||
|
raise netapp_api.NaRetryableError(message=message, code=code)
|
||||||
|
|
||||||
|
try:
|
||||||
|
return _waiter()
|
||||||
|
except netapp_api.NaRetryableError:
|
||||||
|
msg = _("Transfer state did not reach the expected state. Retries "
|
||||||
|
"exhausted. Aborting.")
|
||||||
|
raise na_utils.NetAppDriverException(msg)
|
||||||
|
|
||||||
def update_snapmirror(self, source_vserver, source_volume,
|
def update_snapmirror(self, source_vserver, source_volume,
|
||||||
destination_vserver, destination_volume):
|
destination_vserver, destination_volume):
|
||||||
|
@ -0,0 +1,7 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
NetApp ONTAP driver `bug #2028857
|
||||||
|
<https://bugs.launchpad.net/cinder/+bug/2028857>`_: Fixed
|
||||||
|
errors that were occuring in the replica failover operation when using
|
||||||
|
ONTAP REST API.
|
Loading…
x
Reference in New Issue
Block a user