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:
parent
0a2265d256
commit
7f44844cc1
@ -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):
|
||||||
|
@ -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'}
|
||||||
|
@ -0,0 +1,3 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- Added snapshot manage/unmanage support to the EMC XtremIO driver.
|
Loading…
Reference in New Issue
Block a user