Add cleanup to create from snap in Manila HNAS driver
Adding a cleanup to method create_from_snapshot in case of a failure. Also, raising the correct exception on backend layer in case of a failure when exporting shares. Change-Id: I86d2c3c5ff5a790868f8362e065df1eb2be8a3ad Closes-Bug: #1613721
This commit is contained in:
parent
a97983dd0e
commit
9d6823b3ea
@ -23,7 +23,7 @@ import six
|
|||||||
|
|
||||||
from manila.common import constants
|
from manila.common import constants
|
||||||
from manila import exception
|
from manila import exception
|
||||||
from manila.i18n import _, _LI, _LW
|
from manila.i18n import _, _LE, _LI, _LW
|
||||||
from manila.share import driver
|
from manila.share import driver
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
LOG = log.getLogger(__name__)
|
||||||
@ -812,13 +812,21 @@ class HitachiHNASDriver(driver.ShareDriver):
|
|||||||
|
|
||||||
self._check_protocol(share['id'], share['share_proto'])
|
self._check_protocol(share['id'], share['share_proto'])
|
||||||
|
|
||||||
if share['share_proto'].lower() == 'nfs':
|
try:
|
||||||
self.hnas.nfs_export_add(share['id'])
|
if share['share_proto'].lower() == 'nfs':
|
||||||
uri = self.hnas_evs_ip + ":" + dest_path
|
self.hnas.nfs_export_add(share['id'])
|
||||||
else:
|
uri = self.hnas_evs_ip + ":" + dest_path
|
||||||
self.hnas.cifs_share_add(share['id'])
|
else:
|
||||||
uri = r'\\%s\%s' % (self.hnas_evs_ip, share['id'])
|
self.hnas.cifs_share_add(share['id'])
|
||||||
return uri
|
uri = r'\\%s\%s' % (self.hnas_evs_ip, share['id'])
|
||||||
|
return uri
|
||||||
|
except exception.HNASBackendException:
|
||||||
|
with excutils.save_and_reraise_exception():
|
||||||
|
msg = _LE('Failed to create share %(share_id)s from snapshot '
|
||||||
|
'%(snap)s.')
|
||||||
|
LOG.exception(msg, {'share_id': share['id'],
|
||||||
|
'snap': snapshot['id']})
|
||||||
|
self.hnas.vvol_delete(share['id'])
|
||||||
|
|
||||||
def _check_protocol(self, share_id, protocol):
|
def _check_protocol(self, share_id, protocol):
|
||||||
if protocol.lower() not in ('nfs', 'cifs'):
|
if protocol.lower() not in ('nfs', 'cifs'):
|
||||||
|
@ -65,7 +65,12 @@ class HNASSSHBackend(object):
|
|||||||
path = '/shares/' + share_id
|
path = '/shares/' + share_id
|
||||||
command = ['nfs-export', 'add', '-S', 'disable', '-c', '127.0.0.1',
|
command = ['nfs-export', 'add', '-S', 'disable', '-c', '127.0.0.1',
|
||||||
path, self.fs_name, path]
|
path, self.fs_name, path]
|
||||||
self._execute(command)
|
try:
|
||||||
|
self._execute(command)
|
||||||
|
except processutils.ProcessExecutionError:
|
||||||
|
msg = _("Could not create NFS export %s.") % share_id
|
||||||
|
LOG.exception(msg)
|
||||||
|
raise exception.HNASBackendException(msg=msg)
|
||||||
|
|
||||||
def nfs_export_del(self, share_id):
|
def nfs_export_del(self, share_id):
|
||||||
path = '/shares/' + share_id
|
path = '/shares/' + share_id
|
||||||
@ -85,7 +90,12 @@ class HNASSSHBackend(object):
|
|||||||
path = r'\\shares\\' + share_id
|
path = r'\\shares\\' + share_id
|
||||||
command = ['cifs-share', 'add', '-S', 'disable', '--enable-abe',
|
command = ['cifs-share', 'add', '-S', 'disable', '--enable-abe',
|
||||||
'--nodefaultsaa', share_id, self.fs_name, path]
|
'--nodefaultsaa', share_id, self.fs_name, path]
|
||||||
self._execute(command)
|
try:
|
||||||
|
self._execute(command)
|
||||||
|
except processutils.ProcessExecutionError:
|
||||||
|
msg = _("Could not create CIFS share %s.") % share_id
|
||||||
|
LOG.exception(msg)
|
||||||
|
raise exception.HNASBackendException(msg=msg)
|
||||||
|
|
||||||
def cifs_share_del(self, share_id):
|
def cifs_share_del(self, share_id):
|
||||||
command = ['cifs-share', 'del', '--target-label', self.fs_name,
|
command = ['cifs-share', 'del', '--target-label', self.fs_name,
|
||||||
|
@ -707,6 +707,35 @@ class HitachiHNASTestCase(test.TestCase):
|
|||||||
'context', invalid_share, snapshot_nfs)
|
'context', invalid_share, snapshot_nfs)
|
||||||
self.assertEqual(invalid_protocol_msg, ex.msg)
|
self.assertEqual(invalid_protocol_msg, ex.msg)
|
||||||
|
|
||||||
|
def test_create_share_from_snapshot_cleanup(self):
|
||||||
|
dest_path = '/snapshots/' + share_nfs['id'] + '/' + snapshot_nfs['id']
|
||||||
|
src_path = '/shares/' + share_nfs['id']
|
||||||
|
|
||||||
|
self.mock_object(driver.HitachiHNASDriver, "_check_fs_mounted",
|
||||||
|
mock.Mock())
|
||||||
|
self.mock_object(ssh.HNASSSHBackend, "vvol_create")
|
||||||
|
self.mock_object(ssh.HNASSSHBackend, "quota_add")
|
||||||
|
self.mock_object(ssh.HNASSSHBackend, "tree_clone")
|
||||||
|
self.mock_object(ssh.HNASSSHBackend, "vvol_delete")
|
||||||
|
self.mock_object(ssh.HNASSSHBackend, "nfs_export_add", mock.Mock(
|
||||||
|
side_effect=exception.HNASBackendException(
|
||||||
|
msg='Error adding nfs export.')))
|
||||||
|
|
||||||
|
self.assertRaises(exception.HNASBackendException,
|
||||||
|
self._driver.create_share_from_snapshot,
|
||||||
|
'context', share_nfs, snapshot_nfs)
|
||||||
|
|
||||||
|
ssh.HNASSSHBackend.vvol_create.assert_called_once_with(
|
||||||
|
share_nfs['id'])
|
||||||
|
ssh.HNASSSHBackend.quota_add.assert_called_once_with(
|
||||||
|
share_nfs['id'], share_nfs['size'])
|
||||||
|
ssh.HNASSSHBackend.tree_clone.assert_called_once_with(
|
||||||
|
dest_path, src_path)
|
||||||
|
ssh.HNASSSHBackend.nfs_export_add.assert_called_once_with(
|
||||||
|
share_nfs['id'])
|
||||||
|
ssh.HNASSSHBackend.vvol_delete.assert_called_once_with(
|
||||||
|
share_nfs['id'])
|
||||||
|
|
||||||
def test__check_fs_mounted(self):
|
def test__check_fs_mounted(self):
|
||||||
self._driver._check_fs_mounted()
|
self._driver._check_fs_mounted()
|
||||||
|
|
||||||
|
@ -545,6 +545,14 @@ class HNASSSHTestCase(test.TestCase):
|
|||||||
|
|
||||||
self._driver_ssh._execute.assert_called_with(fake_nfs_command)
|
self._driver_ssh._execute.assert_called_with(fake_nfs_command)
|
||||||
|
|
||||||
|
def test_nfs_export_add_error(self):
|
||||||
|
self.mock_object(ssh.HNASSSHBackend, '_execute', mock.Mock(
|
||||||
|
side_effect=[putils.ProcessExecutionError(stderr='')]))
|
||||||
|
|
||||||
|
self.assertRaises(exception.HNASBackendException,
|
||||||
|
self._driver_ssh.nfs_export_add, 'vvol_test')
|
||||||
|
self.assertTrue(self.mock_log.exception.called)
|
||||||
|
|
||||||
def test_nfs_export_del(self):
|
def test_nfs_export_del(self):
|
||||||
fake_nfs_command = ['nfs-export', 'del', '/shares/vvol_test']
|
fake_nfs_command = ['nfs-export', 'del', '/shares/vvol_test']
|
||||||
self.mock_object(ssh.HNASSSHBackend, '_execute', mock.Mock())
|
self.mock_object(ssh.HNASSSHBackend, '_execute', mock.Mock())
|
||||||
@ -581,6 +589,14 @@ class HNASSSHTestCase(test.TestCase):
|
|||||||
|
|
||||||
self._driver_ssh._execute.assert_called_with(fake_cifs_add_command)
|
self._driver_ssh._execute.assert_called_with(fake_cifs_add_command)
|
||||||
|
|
||||||
|
def test_cifs_share_add_error(self):
|
||||||
|
self.mock_object(ssh.HNASSSHBackend, '_execute', mock.Mock(
|
||||||
|
side_effect=[putils.ProcessExecutionError(stderr='')]))
|
||||||
|
|
||||||
|
self.assertRaises(exception.HNASBackendException,
|
||||||
|
self._driver_ssh.cifs_share_add, 'vvol_test')
|
||||||
|
self.assertTrue(self.mock_log.exception.called)
|
||||||
|
|
||||||
def test_cifs_share_del(self):
|
def test_cifs_share_del(self):
|
||||||
fake_cifs_del_command = ['cifs-share', 'del', '--target-label',
|
fake_cifs_del_command = ['cifs-share', 'del', '--target-label',
|
||||||
self.fs_name, 'vvol_test']
|
self.fs_name, 'vvol_test']
|
||||||
|
@ -0,0 +1,4 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- Fixed Hitachi HNAS driver not cleaning up data in backend when failing
|
||||||
|
to create a share from snapshot.
|
Loading…
Reference in New Issue
Block a user