XtremIO: support of snapshot manage commands

As snapshots are volumes in XtremIO, all this change does is route
the snapshot manage calls to the appropriate volume manage calls.

DocImpact
Change-Id: I90432ace52d9c8b7ae48c843be86c1d51e8c875e
Implements: blueprint xtremio-manage-snapshot
This commit is contained in:
Shay Halsband 2016-07-07 17:17:37 +03:00
parent 0a2265d256
commit 7f44844cc1
3 changed files with 92 additions and 8 deletions

View File

@ -673,6 +673,65 @@ class EMCXIODriverISCSITestCase(BaseEMCXIODriverTestCase):
self.assertRaises(exception.VolumeNotFound, self.driver.unmanage, self.assertRaises(exception.VolumeNotFound, self.driver.unmanage,
self.data.test_volume2) self.data.test_volume2)
def test_manage_snapshot(self, req):
req.side_effect = xms_request
vol_uid = self.data.test_snapshot.volume_id
xms_data['volumes'] = {1: {'name': vol_uid,
'index': 1,
'vol-size': '3',
},
2: {'name': 'unmanaged',
'index': 2,
'ancestor-vol-id': ['', vol_uid, 1],
'vol-size': '3'}
}
ref_vol = {"source-name": "unmanaged"}
self.driver.manage_existing_snapshot(self.data.test_snapshot, ref_vol)
def test_get_manage_snapshot_size(self, req):
req.side_effect = xms_request
vol_uid = self.data.test_snapshot.volume_id
xms_data['volumes'] = {1: {'name': vol_uid,
'index': 1,
'vol-size': '3',
},
2: {'name': 'unmanaged',
'index': 2,
'ancestor-vol-id': ['', vol_uid, 1],
'vol-size': '3'}
}
ref_vol = {"source-name": "unmanaged"}
self.driver.manage_existing_snapshot_get_size(self.data.test_snapshot,
ref_vol)
def test_manage_snapshot_invalid_snapshot(self, req):
req.side_effect = xms_request
xms_data['volumes'] = {1: {'name': 'unmanaged1',
'index': 1,
'vol-size': '3',
'ancestor-vol-id': []}
}
ref_vol = {"source-name": "unmanaged1"}
self.assertRaises(exception.ManageExistingInvalidReference,
self.driver.manage_existing_snapshot,
self.data.test_snapshot, ref_vol)
def test_unmanage_snapshot(self, req):
req.side_effect = xms_request
vol_uid = self.data.test_snapshot.volume_id
xms_data['volumes'] = {1: {'name': vol_uid,
'index': 1,
'vol-size': '3',
},
2: {'name': 'unmanaged',
'index': 2,
'ancestor-vol-id': ['', vol_uid, 1],
'vol-size': '3'}
}
ref_vol = {"source-name": "unmanaged"}
self.driver.manage_existing_snapshot(self.data.test_snapshot, ref_vol)
self.driver.unmanage_snapshot(self.data.test_snapshot)
# ##### Consistancy Groups ##### # ##### Consistancy Groups #####
@mock.patch('cinder.objects.snapshot.SnapshotList.get_all_for_cgsnapshot') @mock.patch('cinder.objects.snapshot.SnapshotList.get_all_for_cgsnapshot')
def test_cg_create(self, get_all_for_cgsnapshot, req): def test_cg_create(self, get_all_for_cgsnapshot, req):

View File

@ -502,22 +502,33 @@ class XtremIOVolumeDriver(san.SanDriver):
self._update_volume_stats() self._update_volume_stats()
return self._stats return self._stats
def manage_existing(self, volume, existing_ref): def manage_existing(self, volume, existing_ref, is_snapshot=False):
"""Manages an existing LV.""" """Manages an existing LV."""
lv_name = existing_ref['source-name'] lv_name = existing_ref['source-name']
# Attempt to locate the volume. # Attempt to locate the volume.
try: try:
vol_obj = self.client.req('volumes', name=lv_name)['content'] vol_obj = self.client.req('volumes', name=lv_name)['content']
if (
is_snapshot and
(not vol_obj['ancestor-vol-id'] or
vol_obj['ancestor-vol-id'][XTREMIO_OID_NAME] !=
volume.volume_id)):
kwargs = {'existing_ref': lv_name,
'reason': 'Not a snapshot of vol %s' %
volume.volume_id}
raise exception.ManageExistingInvalidReference(**kwargs)
except exception.NotFound: except exception.NotFound:
kwargs = {'existing_ref': lv_name, kwargs = {'existing_ref': lv_name,
'reason': 'Specified logical volume does not exist.'} 'reason': 'Specified logical %s does not exist.' %
'snapshot' if is_snapshot else 'volume'}
raise exception.ManageExistingInvalidReference(**kwargs) raise exception.ManageExistingInvalidReference(**kwargs)
# Attempt to rename the LV to match the OpenStack internal name. # Attempt to rename the LV to match the OpenStack internal name.
self.client.req('volumes', 'PUT', data={'vol-name': volume['id']}, self.client.req('volumes', 'PUT', data={'vol-name': volume['id']},
idx=vol_obj['index']) idx=vol_obj['index'])
def manage_existing_get_size(self, volume, existing_ref): def manage_existing_get_size(self, volume, existing_ref,
is_snapshot=False):
"""Return size of an existing LV for manage_existing.""" """Return size of an existing LV for manage_existing."""
# Check that the reference is valid # Check that the reference is valid
if 'source-name' not in existing_ref: if 'source-name' not in existing_ref:
@ -530,7 +541,8 @@ class XtremIOVolumeDriver(san.SanDriver):
vol_obj = self.client.req('volumes', name=lv_name)['content'] vol_obj = self.client.req('volumes', name=lv_name)['content']
except exception.NotFound: except exception.NotFound:
kwargs = {'existing_ref': lv_name, kwargs = {'existing_ref': lv_name,
'reason': 'Specified logical volume does not exist.'} 'reason': 'Specified logical %s does not exist.' %
'snapshot' if is_snapshot else 'volume'}
raise exception.ManageExistingInvalidReference(**kwargs) raise exception.ManageExistingInvalidReference(**kwargs)
# LV size is returned in gigabytes. Attempt to parse size as a float # LV size is returned in gigabytes. Attempt to parse size as a float
# and round up to the next integer. # and round up to the next integer.
@ -538,18 +550,28 @@ class XtremIOVolumeDriver(san.SanDriver):
return lv_size return lv_size
def unmanage(self, volume): def unmanage(self, volume, is_snapshot=False):
"""Removes the specified volume from Cinder management.""" """Removes the specified volume from Cinder management."""
# trying to rename the volume to [cinder name]-unmanged # trying to rename the volume to [cinder name]-unmanged
try: try:
self.client.req('volumes', 'PUT', name=volume['id'], self.client.req('volumes', 'PUT', name=volume['id'],
data={'vol-name': volume['name'] + '-unmanged'}) data={'vol-name': volume['name'] + '-unmanged'})
except exception.NotFound: except exception.NotFound:
LOG.info(_LI("Volume with the name %s wasn't found," LOG.info(_LI("%(typ)s with the name %(name)s wasn't found, "
" can't unmanage"), "can't unmanage") %
volume['id']) {'typ': 'Snapshot' if is_snapshot else 'Volume',
'name': volume['id']})
raise exception.VolumeNotFound(volume_id=volume['id']) raise exception.VolumeNotFound(volume_id=volume['id'])
def manage_existing_snapshot(self, snapshot, existing_ref):
self.manage_existing(snapshot, existing_ref, True)
def manage_existing_snapshot_get_size(self, snapshot, existing_ref):
return self.manage_existing_get_size(snapshot, existing_ref, True)
def unmanage_snapshot(self, snapshot):
self.unmanage(snapshot, True)
def extend_volume(self, volume, new_size): def extend_volume(self, volume, new_size):
"""Extend an existing volume's size.""" """Extend an existing volume's size."""
data = {'vol-size': six.text_type(new_size) + 'g'} data = {'vol-size': six.text_type(new_size) + 'g'}

View File

@ -0,0 +1,3 @@
---
features:
- Added snapshot manage/unmanage support to the EMC XtremIO driver.