Manage/unmanage snapshot in ScaleIO driver

Add support for manage/unmanage snapshot in the ScaleIO driver.

Change-Id: I1b6ff49294977bf086213355c240640117338dab
DocImpact:
Implements: blueprint scaleio-manage-existing-snapshot
This commit is contained in:
Matan Sabag 2016-05-16 04:18:23 -07:00
parent c1c4533757
commit 1861ed5836
3 changed files with 208 additions and 13 deletions

View File

@ -0,0 +1,154 @@
# Copyright (c) 2016 EMC Corporation.
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
from mock import patch
from cinder import context
from cinder import exception
from cinder.tests.unit import fake_constants as fake
from cinder.tests.unit import fake_snapshot
from cinder.tests.unit import fake_volume
from cinder.tests.unit.volume.drivers.emc import scaleio
from cinder.tests.unit.volume.drivers.emc.scaleio import mocks
from cinder.volume import volume_types
class TestManageExistingSnapshot(scaleio.TestScaleIODriver):
"""Test cases for ``ScaleIODriver.manage_existing_snapshot()``"""
def setUp(self):
"""Setup a test case environment.
Creates a fake volume object and sets up the required API responses.
"""
super(TestManageExistingSnapshot, self).setUp()
ctx = context.RequestContext('fake', 'fake', auth_token=True)
self.volume = fake_volume.fake_volume_obj(
ctx, **{'provider_id': fake.PROVIDER_ID})
self.snapshot = fake_snapshot.fake_snapshot_obj(
ctx, **{'provider_id': fake.PROVIDER2_ID})
self.snapshot2 = fake_snapshot.fake_snapshot_obj(
ctx, **{'provider_id': fake.PROVIDER3_ID})
self.snapshot.volume = self.snapshot2.volume = self.volume
self.snapshot['volume_type_id'] = fake.VOLUME_TYPE_ID
self.snapshot2['volume_type_id'] = fake.VOLUME_TYPE_ID
self.snapshot_attached = fake_snapshot.fake_snapshot_obj(
ctx, **{'provider_id': fake.PROVIDER3_ID})
self.HTTPS_MOCK_RESPONSES = {
self.RESPONSE_MODE.Valid: {
'instances/Volume::' + self.volume['provider_id']:
mocks.MockHTTPSResponse({
'id': fake.PROVIDER_ID,
'sizeInKb': 8388608,
'mappedSdcInfo': None,
'ancestorVolumeId': None
}, 200),
'instances/Volume::' + self.snapshot['provider_id']:
mocks.MockHTTPSResponse({
'id': fake.PROVIDER2_ID,
'sizeInKb': 8388608,
'mappedSdcInfo': None,
'ancestorVolumeId': fake.PROVIDER_ID
}, 200),
'instances/Volume::' + self.snapshot2['provider_id']:
mocks.MockHTTPSResponse({
'id': fake.PROVIDER3_ID,
'sizeInKb': 8388608,
'mappedSdcInfo': None,
'ancestorVolumeId': fake.PROVIDER2_ID
}, 200)
},
self.RESPONSE_MODE.BadStatus: {
'instances/Volume::' + self.snapshot['provider_id']:
mocks.MockHTTPSResponse({
'errorCode': 401,
'message': 'BadStatus Volume Test',
}, 401),
'instances/Volume::' + self.snapshot2['provider_id']:
mocks.MockHTTPSResponse({
'id': fake.PROVIDER3_ID,
'sizeInKb': 8388608,
'ancestorVolumeId': fake.PROVIDER2_ID
}, 200),
'instances/Volume::' + self.snapshot_attached['provider_id']:
mocks.MockHTTPSResponse({
'id': fake.PROVIDER3_ID,
'sizeInKb': 8388608,
'mappedSdcInfo': 'Mapped',
'ancestorVolumeId': fake.PROVIDER_ID
}, 200)
}
}
def test_no_source_id(self):
existing_ref = {'source-name': 'scaleioSnapName'}
self.assertRaises(exception.ManageExistingInvalidReference,
self.driver.manage_existing_snapshot, self.snapshot,
existing_ref)
@patch.object(
volume_types,
'get_volume_type',
return_value={'extra_specs': {'volume_backend_name': 'ScaleIO'}})
def test_snapshot_not_found(self, _mock_volume_type):
existing_ref = {'source-id': fake.PROVIDER2_ID}
self.set_https_response_mode(self.RESPONSE_MODE.BadStatus)
self.assertRaises(exception.ManageExistingInvalidReference,
self.driver.manage_existing_snapshot, self.snapshot,
existing_ref)
@patch.object(
volume_types,
'get_volume_type',
return_value={'extra_specs': {'volume_backend_name': 'ScaleIO'}})
def test_snapshot_attached(self, _mock_volume_type):
self.snapshot_attached['volume_type_id'] = fake.VOLUME_TYPE_ID
existing_ref = {'source-id': fake.PROVIDER2_ID}
self.set_https_response_mode(self.RESPONSE_MODE.BadStatus)
self.assertRaises(exception.ManageExistingInvalidReference,
self.driver.manage_existing_snapshot,
self.snapshot_attached, existing_ref)
@patch.object(
volume_types,
'get_volume_type',
return_value={'extra_specs': {'volume_backend_name': 'ScaleIO'}})
def test_different_ancestor(self, _mock_volume_type):
existing_ref = {'source-id': fake.PROVIDER3_ID}
self.set_https_response_mode(self.RESPONSE_MODE.Valid)
self.assertRaises(exception.ManageExistingInvalidReference,
self.driver.manage_existing_snapshot,
self.snapshot2, existing_ref)
@patch.object(
volume_types,
'get_volume_type',
return_value={'extra_specs': {'volume_backend_name': 'ScaleIO'}})
def test_manage_snapshot_get_size_calc(self, _mock_volume_type):
existing_ref = {'source-id': fake.PROVIDER2_ID}
self.set_https_response_mode(self.RESPONSE_MODE.Valid)
result = self.driver.manage_existing_snapshot_get_size(
self.snapshot, existing_ref)
self.assertEqual(8, result)
@patch.object(
volume_types,
'get_volume_type',
return_value={'extra_specs': {'volume_backend_name': 'ScaleIO'}})
def test_manage_existing_snapshot_valid(self, _mock_volume_type):
existing_ref = {'source-id': fake.PROVIDER2_ID}
result = self.driver.manage_existing_snapshot(
self.snapshot, existing_ref)
self.assertEqual(fake.PROVIDER2_ID, result['provider_id'])

View File

@ -1022,31 +1022,60 @@ class ScaleIODriver(driver.VolumeDriver):
"%(new_name)s."), "%(new_name)s."),
{'vol': vol_id, 'new_name': new_name}) {'vol': vol_id, 'new_name': new_name})
def _query_scaleio_volume(self, volume, existing_ref):
request = self._create_scaleio_get_volume_request(volume, existing_ref)
r, response = self._execute_scaleio_get_request(request)
LOG.info(_LI("Get Volume response: %(res)s"),
{'res': response})
self._manage_existing_check_legal_response(r, existing_ref)
return response
def manage_existing(self, volume, existing_ref): def manage_existing(self, volume, existing_ref):
"""Manage an existing ScaleIO volume. """Manage an existing ScaleIO volume.
existing_ref is a dictionary of the form: existing_ref is a dictionary of the form:
{'source-id': <id of ScaleIO volume>} {'source-id': <id of ScaleIO volume>}
""" """
request = self._create_scaleio_get_volume_request(volume, existing_ref) response = self._query_scaleio_volume(volume, existing_ref)
r, response = self._execute_scaleio_get_request(request) return {'provider_id': response['id']}
LOG.info(_LI("Get Volume response: %s"), response)
self._manage_existing_check_legal_response(r, existing_ref) def manage_existing_get_size(self, volume, existing_ref):
if response['mappedSdcInfo'] is not None: return self._get_volume_size(volume, existing_ref)
reason = _("manage_existing cannot manage a volume "
"connected to hosts. Please disconnect this volume " def manage_existing_snapshot(self, snapshot, existing_ref):
"from existing hosts before importing") """Manage an existing ScaleIO snapshot.
:param existing_ref: dictionary of the form:
{'source-id': <id of ScaleIO snapshot>}
"""
response = self._query_scaleio_volume(snapshot, existing_ref)
not_real_parent = (response.get('orig_parent_overriden') or
response.get('is_source_deleted'))
if not_real_parent:
reason = (_("The snapshot's parent is not the original parent due "
"to deletion or revert action, therefore "
"this snapshot cannot be managed."))
raise exception.ManageExistingInvalidReference(
existing_ref=existing_ref,
reason=reason
)
ancestor_id = response['ancestorVolumeId']
volume_id = snapshot.volume.provider_id
if ancestor_id != volume_id:
reason = (_("The snapshot's parent in ScaleIO is %(ancestor)s "
"and not %(volume)s.") %
{'ancestor': ancestor_id, 'volume': volume_id})
raise exception.ManageExistingInvalidReference( raise exception.ManageExistingInvalidReference(
existing_ref=existing_ref, existing_ref=existing_ref,
reason=reason reason=reason
) )
return {'provider_id': response['id']} return {'provider_id': response['id']}
def manage_existing_get_size(self, volume, existing_ref): def manage_existing_snapshot_get_size(self, snapshot, existing_ref):
request = self._create_scaleio_get_volume_request(volume, existing_ref) return self._get_volume_size(snapshot, existing_ref)
r, response = self._execute_scaleio_get_request(request)
LOG.info(_LI("Get Volume response: %s"), response) def _get_volume_size(self, volume, existing_ref):
self._manage_existing_check_legal_response(r, existing_ref) response = self._query_scaleio_volume(volume, existing_ref)
return int(response['sizeInKb'] / units.Mi) return int(response['sizeInKb'] / units.Mi)
def _execute_scaleio_get_request(self, request): def _execute_scaleio_get_request(self, request):
@ -1096,6 +1125,15 @@ class ScaleIODriver(driver.VolumeDriver):
reason=reason reason=reason
) )
if response.json()['mappedSdcInfo'] is not None:
reason = _("manage_existing cannot manage a volume "
"connected to hosts. Please disconnect this volume "
"from existing hosts before importing.")
raise exception.ManageExistingInvalidReference(
existing_ref=existing_ref,
reason=reason
)
def create_consistencygroup(self, context, group): def create_consistencygroup(self, context, group):
"""Creates a consistency group. """Creates a consistency group.

View File

@ -0,0 +1,3 @@
---
features:
- Added support for manage/unmanage snapshot in the ScaleIO driver.