2014-07-02 14:38:47 +00:00
|
|
|
|
|
|
|
from StringIO import StringIO
|
2014-07-02 18:25:14 +00:00
|
|
|
import json
|
2014-07-02 14:38:47 +00:00
|
|
|
import time
|
|
|
|
import os
|
|
|
|
import logging
|
|
|
|
|
|
|
|
from teuthology import misc
|
2014-08-07 14:24:59 +00:00
|
|
|
from teuthology.orchestra import run
|
2014-07-02 18:25:14 +00:00
|
|
|
from teuthology.orchestra.run import CommandFailedError
|
2014-08-07 14:24:59 +00:00
|
|
|
from .mount import CephFSMount
|
2014-07-02 14:38:47 +00:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2014-07-02 15:43:16 +00:00
|
|
|
class FuseMount(CephFSMount):
|
2014-07-02 14:38:47 +00:00
|
|
|
def __init__(self, client_config, test_dir, client_id, client_remote):
|
2014-07-02 15:43:16 +00:00
|
|
|
super(FuseMount, self).__init__(test_dir, client_id, client_remote)
|
2014-07-02 14:38:47 +00:00
|
|
|
|
2014-07-02 18:25:14 +00:00
|
|
|
self.client_config = client_config if client_config else {}
|
2014-07-02 14:38:47 +00:00
|
|
|
self.fuse_daemon = None
|
|
|
|
|
|
|
|
def mount(self):
|
|
|
|
log.info("Client client.%s config is %s" % (self.client_id, self.client_config))
|
|
|
|
|
|
|
|
daemon_signal = 'kill'
|
|
|
|
if self.client_config.get('coverage') or self.client_config.get('valgrind') is not None:
|
|
|
|
daemon_signal = 'term'
|
|
|
|
|
|
|
|
mnt = os.path.join(self.test_dir, 'mnt.{id}'.format(id=self.client_id))
|
|
|
|
log.info('Mounting ceph-fuse client.{id} at {remote} {mnt}...'.format(
|
|
|
|
id=self.client_id, remote=self.client_remote, mnt=mnt))
|
|
|
|
|
|
|
|
self.client_remote.run(
|
|
|
|
args=[
|
|
|
|
'mkdir',
|
|
|
|
'--',
|
|
|
|
mnt,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
|
|
|
run_cmd = [
|
|
|
|
'sudo',
|
|
|
|
'adjust-ulimits',
|
|
|
|
'ceph-coverage',
|
|
|
|
'{tdir}/archive/coverage'.format(tdir=self.test_dir),
|
|
|
|
'daemon-helper',
|
|
|
|
daemon_signal,
|
|
|
|
]
|
|
|
|
run_cmd_tail = [
|
|
|
|
'ceph-fuse',
|
|
|
|
'-f',
|
|
|
|
'--name', 'client.{id}'.format(id=self.client_id),
|
|
|
|
# TODO ceph-fuse doesn't understand dash dash '--',
|
|
|
|
mnt,
|
|
|
|
]
|
|
|
|
|
|
|
|
if self.client_config.get('valgrind') is not None:
|
|
|
|
run_cmd = misc.get_valgrind_args(
|
|
|
|
self.test_dir,
|
|
|
|
'client.{id}'.format(id=self.client_id),
|
|
|
|
run_cmd,
|
|
|
|
self.client_config.get('valgrind'),
|
|
|
|
)
|
|
|
|
|
|
|
|
run_cmd.extend(run_cmd_tail)
|
|
|
|
|
|
|
|
proc = self.client_remote.run(
|
|
|
|
args=run_cmd,
|
|
|
|
logger=log.getChild('ceph-fuse.{id}'.format(id=self.client_id)),
|
|
|
|
stdin=run.PIPE,
|
|
|
|
wait=False,
|
|
|
|
)
|
|
|
|
self.fuse_daemon = proc
|
|
|
|
|
|
|
|
def is_mounted(self):
|
2014-08-20 11:36:02 +00:00
|
|
|
try:
|
|
|
|
proc = self.client_remote.run(
|
|
|
|
args=[
|
|
|
|
'stat',
|
|
|
|
'--file-system',
|
|
|
|
'--printf=%T\n',
|
|
|
|
'--',
|
|
|
|
self.mountpoint,
|
|
|
|
],
|
|
|
|
stdout=StringIO(),
|
|
|
|
)
|
|
|
|
except CommandFailedError:
|
|
|
|
# This happens if the mount directory doesn't exist
|
2014-10-02 12:01:56 +00:00
|
|
|
log.info('mount point does not exist: %s', self.mountpoint)
|
2014-08-20 11:36:02 +00:00
|
|
|
return False
|
|
|
|
|
2014-07-02 14:38:47 +00:00
|
|
|
fstype = proc.stdout.getvalue().rstrip('\n')
|
|
|
|
if fstype == 'fuseblk':
|
|
|
|
log.info('ceph-fuse is mounted on %s', self.mountpoint)
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
log.debug('ceph-fuse not mounted, got fs type {fstype!r}'.format(
|
|
|
|
fstype=fstype))
|
2015-02-10 12:37:20 +00:00
|
|
|
return False
|
2014-07-02 14:38:47 +00:00
|
|
|
|
|
|
|
def wait_until_mounted(self):
|
|
|
|
"""
|
|
|
|
Check to make sure that fuse is mounted on mountpoint. If not,
|
|
|
|
sleep for 5 seconds and check again.
|
|
|
|
"""
|
|
|
|
|
|
|
|
while not self.is_mounted():
|
|
|
|
# Even if it's not mounted, it should at least
|
|
|
|
# be running: catch simple failures where it has terminated.
|
|
|
|
assert not self.fuse_daemon.poll()
|
|
|
|
|
|
|
|
time.sleep(5)
|
|
|
|
|
|
|
|
# Now that we're mounted, set permissions so that the rest of the test will have
|
|
|
|
# unrestricted access to the filesystem mount.
|
|
|
|
self.client_remote.run(
|
2014-09-15 22:41:34 +00:00
|
|
|
args=['sudo', 'chmod', '1777', self.mountpoint])
|
2014-07-02 14:38:47 +00:00
|
|
|
|
2014-08-20 11:36:02 +00:00
|
|
|
def _mountpoint_exists(self):
|
|
|
|
return self.client_remote.run(args=["ls", "-d", self.mountpoint], check_status=False).exitstatus == 0
|
|
|
|
|
2014-07-02 14:38:47 +00:00
|
|
|
def umount(self):
|
|
|
|
try:
|
2014-09-03 13:15:54 +00:00
|
|
|
log.info('Running fusermount -u on {name}...'.format(name=self.client_remote.name))
|
2014-07-02 14:38:47 +00:00
|
|
|
self.client_remote.run(
|
|
|
|
args=[
|
|
|
|
'sudo',
|
|
|
|
'fusermount',
|
|
|
|
'-u',
|
2014-07-02 18:25:14 +00:00
|
|
|
self.mountpoint,
|
2014-07-02 14:38:47 +00:00
|
|
|
],
|
|
|
|
)
|
|
|
|
except run.CommandFailedError:
|
|
|
|
log.info('Failed to unmount ceph-fuse on {name}, aborting...'.format(name=self.client_remote.name))
|
2015-02-10 12:37:20 +00:00
|
|
|
|
2014-07-02 14:38:47 +00:00
|
|
|
# abort the fuse mount, killing all hung processes
|
|
|
|
self.client_remote.run(
|
|
|
|
args=[
|
2015-02-10 12:37:20 +00:00
|
|
|
"find", "/sys/fs/fuse/connections", "-name", "abort",
|
|
|
|
"-exec", "bash", "-c", "echo 1 > {}", "\;"
|
|
|
|
]
|
2014-07-02 14:38:47 +00:00
|
|
|
)
|
|
|
|
# make sure its unmounted
|
2015-02-10 12:37:20 +00:00
|
|
|
if self.is_mounted():
|
2014-08-20 11:36:02 +00:00
|
|
|
self.client_remote.run(
|
|
|
|
args=[
|
|
|
|
'sudo',
|
|
|
|
'umount',
|
|
|
|
'-l',
|
|
|
|
'-f',
|
|
|
|
self.mountpoint,
|
|
|
|
],
|
|
|
|
)
|
2014-07-02 14:38:47 +00:00
|
|
|
|
2014-07-02 18:25:14 +00:00
|
|
|
def umount_wait(self, force=False):
|
|
|
|
"""
|
|
|
|
:param force: Complete even if the MDS is offline
|
|
|
|
"""
|
|
|
|
self.umount()
|
|
|
|
if force:
|
|
|
|
self.fuse_daemon.stdin.close()
|
|
|
|
try:
|
|
|
|
self.fuse_daemon.wait()
|
|
|
|
except CommandFailedError:
|
|
|
|
pass
|
|
|
|
self.cleanup()
|
|
|
|
|
2014-07-02 14:38:47 +00:00
|
|
|
def cleanup(self):
|
|
|
|
"""
|
|
|
|
Remove the mount point.
|
|
|
|
|
|
|
|
Prerequisite: the client is not mounted.
|
|
|
|
"""
|
|
|
|
self.client_remote.run(
|
|
|
|
args=[
|
|
|
|
'rmdir',
|
|
|
|
'--',
|
2014-07-02 18:25:14 +00:00
|
|
|
self.mountpoint,
|
2014-07-02 14:38:47 +00:00
|
|
|
],
|
|
|
|
)
|
2014-07-02 18:25:14 +00:00
|
|
|
|
|
|
|
def kill(self):
|
|
|
|
"""
|
|
|
|
Terminate the client without removing the mount point.
|
|
|
|
"""
|
|
|
|
self.fuse_daemon.stdin.close()
|
|
|
|
try:
|
|
|
|
self.fuse_daemon.wait()
|
|
|
|
except CommandFailedError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def kill_cleanup(self):
|
|
|
|
"""
|
|
|
|
Follow up ``kill`` to get to a clean unmounted state.
|
|
|
|
"""
|
|
|
|
self.umount()
|
|
|
|
self.cleanup()
|
|
|
|
|
|
|
|
def teardown(self):
|
|
|
|
"""
|
|
|
|
Whatever the state of the mount, get it gone.
|
|
|
|
"""
|
|
|
|
super(FuseMount, self).teardown()
|
|
|
|
|
2014-08-20 11:36:02 +00:00
|
|
|
if self.is_mounted():
|
|
|
|
self.umount()
|
2014-07-02 18:25:14 +00:00
|
|
|
if not self.fuse_daemon.finished:
|
|
|
|
self.fuse_daemon.stdin.close()
|
|
|
|
try:
|
|
|
|
self.fuse_daemon.wait()
|
|
|
|
except CommandFailedError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Indiscriminate, unlike the touchier cleanup()
|
|
|
|
self.client_remote.run(
|
|
|
|
args=[
|
|
|
|
'rm',
|
|
|
|
'-rf',
|
|
|
|
self.mountpoint,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
2014-10-28 17:12:02 +00:00
|
|
|
def _admin_socket(self, args):
|
2014-07-02 18:25:14 +00:00
|
|
|
pyscript = """
|
|
|
|
import glob
|
|
|
|
import re
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
def find_socket(client_name):
|
|
|
|
files = glob.glob("/var/run/ceph/ceph-{{client_name}}.*.asok".format(client_name=client_name))
|
|
|
|
for f in files:
|
|
|
|
pid = re.match(".*\.(\d+)\.asok$", f).group(1)
|
|
|
|
if os.path.exists("/proc/{{0}}".format(pid)):
|
|
|
|
return f
|
|
|
|
raise RuntimeError("Client socket {{0}} not found".format(client_name))
|
|
|
|
|
|
|
|
print find_socket("{client_name}")
|
|
|
|
""".format(client_name="client.{0}".format(self.client_id))
|
|
|
|
|
|
|
|
# Find the admin socket
|
|
|
|
p = self.client_remote.run(args=[
|
|
|
|
'python', '-c', pyscript
|
|
|
|
], stdout=StringIO())
|
|
|
|
asok_path = p.stdout.getvalue().strip()
|
|
|
|
log.info("Found client admin socket at {0}".format(asok_path))
|
|
|
|
|
|
|
|
# Query client ID from admin socket
|
|
|
|
p = self.client_remote.run(
|
2014-10-28 17:12:02 +00:00
|
|
|
args=['sudo', 'ceph', '--admin-daemon', asok_path] + args,
|
2014-07-02 18:25:14 +00:00
|
|
|
stdout=StringIO())
|
2014-10-28 17:12:02 +00:00
|
|
|
return json.loads(p.stdout.getvalue())
|
|
|
|
|
|
|
|
def get_global_id(self):
|
|
|
|
"""
|
|
|
|
Look up the CephFS client ID for this mount
|
|
|
|
"""
|
|
|
|
|
|
|
|
return self._admin_socket(['mds_sessions'])['id']
|
|
|
|
|
|
|
|
def get_osd_epoch(self):
|
|
|
|
"""
|
|
|
|
Return 2-tuple of osd_epoch, osd_epoch_barrier
|
|
|
|
"""
|
|
|
|
status = self._admin_socket(['status'])
|
|
|
|
return status['osd_epoch'], status['osd_epoch_barrier']
|