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:
Alyson Rosa 2016-08-16 15:00:24 -03:00
parent a97983dd0e
commit 9d6823b3ea
5 changed files with 77 additions and 10 deletions

View File

@ -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'):

View File

@ -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,

View File

@ -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()

View File

@ -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']

View File

@ -0,0 +1,4 @@
---
fixes:
- Fixed Hitachi HNAS driver not cleaning up data in backend when failing
to create a share from snapshot.