Remove nsenter dependency

Rather complex logic was added to ZFS driver to cope with
mount-namespace-related problems caused by neutron. A better
solution is to ensure mounts (and unmount) propagate to
namespaces.

Closes-Bug: 1552552
Change-Id: Id952cb110c0187021a46ed5de02308b10ddf1239
This commit is contained in:
Ben Swartzlander 2016-03-02 11:53:28 -05:00
parent 71f8d0f3df
commit fad7709ed2
6 changed files with 46 additions and 145 deletions

View File

@ -580,12 +580,6 @@ function install_manila {
sudo apt-get install -y build-essential sudo apt-get install -y build-essential
sudo apt-get install -y ubuntu-zfs sudo apt-get install -y ubuntu-zfs
sudo modprobe zfs sudo modprobe zfs
# TODO(vponomaryov): remove following line when we have this
# in 'requirements.txt' file.
# Package 'nsenter' is expected to be installed on host with
# ZFS, if it is remote for manila-share service host.
sudo pip install nsenter
else else
echo "Manila Devstack plugin does not support installation "\ echo "Manila Devstack plugin does not support installation "\
"of ZFS packages for non-'Ubuntu-trusty' distros. "\ "of ZFS packages for non-'Ubuntu-trusty' distros. "\

View File

@ -146,9 +146,6 @@ zpool: CommandFilter, zpool, root
# manila/share/drivers/zfsonlinux/utils.py # manila/share/drivers/zfsonlinux/utils.py
zfs: CommandFilter, zfs, root zfs: CommandFilter, zfs, root
# manila/share/drivers/zfsonlinux/driver.py
nsenter: CommandFilter, nsenter, root
# LXD driver commands # LXD driver commands
# manila/share/drivers/lxd.py # manila/share/drivers/lxd.py
lxc: CommandFilter, lxc, root lxc: CommandFilter, lxc, root

View File

@ -27,7 +27,7 @@ from oslo_utils import timeutils
from manila.common import constants from manila.common import constants
from manila import exception from manila import exception
from manila.i18n import _, _LW from manila.i18n import _, _LI, _LW
from manila.share import driver from manila.share import driver
from manila.share.drivers.zfsonlinux import utils as zfs_utils from manila.share.drivers.zfsonlinux import utils as zfs_utils
from manila.share import utils as share_utils from manila.share import utils as share_utils
@ -186,46 +186,29 @@ class ZFSonLinuxShareDriver(zfs_utils.ExecuteMixin, driver.ShareDriver):
msg=_("Could not destroy '%s' dataset, " msg=_("Could not destroy '%s' dataset, "
"because it had opened files.") % name) "because it had opened files.") % name)
try:
self.zfs('destroy', '-f', name)
return
except exception.ProcessExecutionError as e:
LOG.debug("Failed to run command, got error: %s\n"
"Assuming other namespace-based services hold "
"ZFS mounts.", e)
# NOTE(vponomaryov): perform workaround for Neutron bug #1546723
# We should release ZFS mount from all namespaces. It should not be
# there at all.
get_pids_cmd = (
"(echo $(grep -s %s /proc/*/mounts) ) 2>&1 " % mountpoint)
try:
raw_pids, err = self.execute('bash', '-c', get_pids_cmd)
except exception.ProcessExecutionError as e:
LOG.warning(
_LW("Failed to get list of PIDs that hold ZFS dataset "
"mountpoint. Got following error: %s"), e)
else:
pids = [s.split('/')[0] for s in raw_pids.split('/proc/') if s]
LOG.debug(
"List of pids that hold ZFS mount '%(mnt)s': %(list)s", {
'mnt': mountpoint, 'list': ' '.join(pids)})
for pid in pids:
try:
self.execute(
'sudo', 'nsenter', '--mnt', '--target=%s' % pid,
'/bin/umount', mountpoint)
except exception.ProcessExecutionError as e:
LOG.warning(
_LW("Failed to run command with release of "
"ZFS dataset mount, got error: %s"), e)
# NOTE(vponomaryov): sleep some time after unmount operations.
time.sleep(1)
# NOTE(vponomaryov): Now, when no file usages and mounts of dataset # NOTE(vponomaryov): Now, when no file usages and mounts of dataset
# exist, destroy dataset. # exist, destroy dataset.
self.zfs_with_retry('destroy', '-f', name) try:
self.zfs('destroy', '-f', name)
return
except exception.ProcessExecutionError:
LOG.info(_LI("Failed to destroy ZFS dataset, retrying one time"))
# NOTE(bswartz): There appears to be a bug in ZFS when creating and
# destroying datasets concurrently where the filesystem remains mounted
# even though ZFS thinks it's unmounted. The most reliable workaround
# I've found is to force the unmount, then retry the destroy, with
# short pauses around the unmount.
time.sleep(1)
try:
self.execute('sudo', 'umount', mountpoint)
except exception.ProcessExecutionError:
# Ignore failed umount, it's normal
pass
time.sleep(1)
# This time the destroy is expected to succeed.
self.zfs('destroy', '-f', name)
def _setup_helpers(self): def _setup_helpers(self):
"""Setups share helper for ZFS backend.""" """Setups share helper for ZFS backend."""
@ -268,6 +251,11 @@ class ZFSonLinuxShareDriver(zfs_utils.ExecuteMixin, driver.ShareDriver):
reason=_("No zpools specified for usage: " reason=_("No zpools specified for usage: "
"%s") % self.zpool_list) "%s") % self.zpool_list)
# Make pool mounts shared so that cloned namespaces receive unmounts
# and don't prevent us from unmounting datasets
for zpool in self.configuration.zfs_zpool_list:
self.execute('sudo', 'mount', '--make-rshared', ('/%s' % zpool))
if self.configuration.zfs_use_ssh: if self.configuration.zfs_use_ssh:
# Check workability of SSH executor # Check workability of SSH executor
self.ssh_executor('whoami') self.ssh_executor('whoami')

View File

@ -124,10 +124,6 @@ class ExecuteMixin(driver.ExecuteMixin):
"""ZFS shell commands executor.""" """ZFS shell commands executor."""
return self.execute('sudo', 'zfs', *cmd, **kwargs) return self.execute('sudo', 'zfs', *cmd, **kwargs)
def zfs_with_retry(self, *cmd, **kwargs):
"""ZFS shell commands executor with retries."""
return self.execute_with_retry('sudo', 'zfs', *cmd, **kwargs)
@six.add_metaclass(abc.ABCMeta) @six.add_metaclass(abc.ABCMeta)
class NASHelperBase(object): class NASHelperBase(object):

View File

@ -121,8 +121,7 @@ class ZFSonLinuxShareDriverTestCase(test.TestCase):
self.assertTrue(hasattr(self.driver, '_helpers')) self.assertTrue(hasattr(self.driver, '_helpers'))
self.assertEqual(self.driver._helpers, {}) self.assertEqual(self.driver._helpers, {})
for attr_name in ('execute', 'execute_with_retry', 'parse_zfs_answer', for attr_name in ('execute', 'execute_with_retry', 'parse_zfs_answer',
'get_zpool_option', 'get_zfs_option', 'zfs', 'get_zpool_option', 'get_zfs_option', 'zfs'):
'zfs_with_retry'):
self.assertTrue(hasattr(self.driver, attr_name)) self.assertTrue(hasattr(self.driver, attr_name))
def test_init_error_with_duplicated_zpools(self): def test_init_error_with_duplicated_zpools(self):
@ -177,9 +176,9 @@ class ZFSonLinuxShareDriverTestCase(test.TestCase):
self.driver._setup_helpers.assert_called_once_with() self.driver._setup_helpers.assert_called_once_with()
if use_ssh: if use_ssh:
self.driver.ssh_executor.assert_called_once_with('whoami') self.assertEqual(4, self.driver.ssh_executor.call_count)
else: else:
self.assertEqual(0, self.driver.ssh_executor.call_count) self.assertEqual(3, self.driver.ssh_executor.call_count)
@ddt.data( @ddt.data(
('foo', '127.0.0.1'), ('foo', '127.0.0.1'),
@ -846,13 +845,13 @@ class ZFSonLinuxShareDriverTestCase(test.TestCase):
def test__delete_dataset_or_snapshot_with_retry_snapshot(self): def test__delete_dataset_or_snapshot_with_retry_snapshot(self):
self.mock_object(self.driver, 'get_zfs_option') self.mock_object(self.driver, 'get_zfs_option')
self.mock_object(self.driver, 'zfs_with_retry') self.mock_object(self.driver, 'zfs')
self.driver._delete_dataset_or_snapshot_with_retry('foo@bar') self.driver._delete_dataset_or_snapshot_with_retry('foo@bar')
self.driver.get_zfs_option.assert_called_once_with( self.driver.get_zfs_option.assert_called_once_with(
'foo@bar', 'mountpoint') 'foo@bar', 'mountpoint')
self.driver.zfs_with_retry.assert_called_once_with( self.driver.zfs.assert_called_once_with(
'destroy', '-f', 'foo@bar') 'destroy', '-f', 'foo@bar')
def test__delete_dataset_or_snapshot_with_retry_of(self): def test__delete_dataset_or_snapshot_with_retry_of(self):
@ -903,92 +902,29 @@ class ZFSonLinuxShareDriverTestCase(test.TestCase):
zfs_driver.time.sleep.assert_called_once_with(2) zfs_driver.time.sleep.assert_called_once_with(2)
self.driver.zfs.assert_called_once_with('destroy', '-f', dataset_name) self.driver.zfs.assert_called_once_with('destroy', '-f', dataset_name)
def test__delete_dataset_or_snapshot_with_retry_mount_holders(self): def test__delete_dataset_or_snapshot_with_retry_busy(self):
mnt = 'fake_mnt' self.mock_object(self.driver, 'get_zfs_option')
self.mock_object( self.mock_object(
self.driver, 'get_zfs_option', mock.Mock(return_value=mnt)) self.driver, 'execute', mock.Mock(
self.mock_object(self.driver, 'zfs_with_retry') side_effect=exception.ProcessExecutionError(
'FAKE lsof returns not found')))
self.mock_object( self.mock_object(
self.driver, 'zfs', self.driver, 'zfs', mock.Mock(side_effect=[
mock.Mock(side_effect=exception.ProcessExecutionError('FAKE'))) exception.ProcessExecutionError(
pids_raw = '/proc/1234/foo /proc/5678/bar' 'cannot destroy FAKE: dataset is busy\n'),
self.mock_object( None, None]))
self.driver, 'execute', mock.Mock(side_effect=[
('a', 'b'),
exception.ProcessExecutionError('Fake lsof empty'),
(pids_raw, 'c'),
exception.ProcessExecutionError('Fake PID not found error'),
('d', 'e'),
]))
self.mock_object(zfs_driver.time, 'sleep') self.mock_object(zfs_driver.time, 'sleep')
self.mock_object(zfs_driver.LOG, 'debug') self.mock_object(zfs_driver.LOG, 'info')
self.mock_object(zfs_driver.LOG, 'warning')
self.mock_object(
zfs_driver.time, 'time', mock.Mock(side_effect=range(1, 70, 2)))
dataset_name = 'fake/dataset/name' dataset_name = 'fake/dataset/name'
self.driver._delete_dataset_or_snapshot_with_retry(dataset_name) self.driver._delete_dataset_or_snapshot_with_retry(dataset_name)
self.driver.get_zfs_option.assert_called_once_with( self.driver.get_zfs_option.assert_called_once_with(
dataset_name, 'mountpoint') dataset_name, 'mountpoint')
self.assertEqual(3, zfs_driver.time.time.call_count) self.assertEqual(2, zfs_driver.time.sleep.call_count)
self.assertEqual(3, zfs_driver.LOG.debug.call_count) self.assertEqual(2, self.driver.execute.call_count)
self.assertEqual(5, self.driver.execute.call_count) self.assertEqual(1, zfs_driver.LOG.info.call_count)
self.driver.execute.assert_has_calls([ self.assertEqual(2, self.driver.zfs.call_count)
mock.call('lsof', '-w', mnt),
mock.call('lsof', '-w', mnt),
mock.call('bash', '-c',
'(echo $(grep -s %s /proc/*/mounts) ) 2>&1 ' % mnt),
mock.call('sudo', 'nsenter', '--mnt', '--target=1234',
'/bin/umount', mnt),
mock.call('sudo', 'nsenter', '--mnt', '--target=5678',
'/bin/umount', mnt),
])
zfs_driver.time.sleep.assert_has_calls([mock.call(2), mock.call(1)])
self.driver.zfs.assert_called_once_with('destroy', '-f', dataset_name)
zfs_driver.LOG.warning.assert_called_once_with(mock.ANY, mock.ANY)
self.driver.zfs_with_retry.asssert_called_once_with(
'destroy', '-f', dataset_name)
def test__delete_dataset_or_snapshot_with_retry_mount_holders_error(self):
mnt = 'fake_mnt'
self.mock_object(
self.driver, 'get_zfs_option', mock.Mock(return_value=mnt))
self.mock_object(self.driver, 'zfs_with_retry')
self.mock_object(
self.driver, 'zfs',
mock.Mock(side_effect=exception.ProcessExecutionError('FAKE')))
self.mock_object(
self.driver, 'execute', mock.Mock(side_effect=[
('a', 'b'),
exception.ProcessExecutionError('Fake lsof empty'),
exception.ProcessExecutionError('Failed to get list of PIDs'),
]))
self.mock_object(zfs_driver.time, 'sleep')
self.mock_object(zfs_driver.LOG, 'debug')
self.mock_object(zfs_driver.LOG, 'warning')
self.mock_object(
zfs_driver.time, 'time', mock.Mock(side_effect=range(1, 70, 2)))
dataset_name = 'fake/dataset/name'
self.driver._delete_dataset_or_snapshot_with_retry(dataset_name)
self.driver.get_zfs_option.assert_called_once_with(
dataset_name, 'mountpoint')
self.assertEqual(3, zfs_driver.time.time.call_count)
self.assertEqual(2, zfs_driver.LOG.debug.call_count)
self.assertEqual(3, self.driver.execute.call_count)
self.driver.execute.assert_has_calls([
mock.call('lsof', '-w', mnt),
mock.call('lsof', '-w', mnt),
mock.call('bash', '-c',
'(echo $(grep -s %s /proc/*/mounts) ) 2>&1 ' % mnt),
])
zfs_driver.time.sleep.assert_has_calls([mock.call(2), mock.call(1)])
self.driver.zfs.assert_called_once_with('destroy', '-f', dataset_name)
zfs_driver.LOG.warning.assert_called_once_with(mock.ANY, mock.ANY)
self.driver.zfs_with_retry.asssert_called_once_with(
'destroy', '-f', dataset_name)
def test_create_replica(self): def test_create_replica(self):
active_replica = { active_replica = {

View File

@ -213,16 +213,6 @@ foo_res opt_3 some_value local"""
self.driver.execute.asssert_called_once_with( self.driver.execute.asssert_called_once_with(
'sudo', 'zfs', 'foo', 'bar') 'sudo', 'zfs', 'foo', 'bar')
def test_zfs_with_retrying(self):
self.mock_object(self.driver, 'execute')
self.mock_object(self.driver, 'execute_with_retry')
self.driver.zfs_with_retry('foo', 'bar')
self.assertEqual(0, self.driver.execute.call_count)
self.driver.execute_with_retry.asssert_called_once_with(
'sudo', 'zfs', 'foo', 'bar')
@ddt.ddt @ddt.ddt
class NFSviaZFSHelperTestCase(test.TestCase): class NFSviaZFSHelperTestCase(test.TestCase):