2014-11-20 06:55:39 +00:00
|
|
|
|
|
|
|
"""
|
|
|
|
Exercise the MDS's auto repair functions
|
|
|
|
"""
|
|
|
|
|
|
|
|
import logging
|
|
|
|
import time
|
|
|
|
|
2021-06-15 12:55:22 +00:00
|
|
|
from teuthology.exceptions import CommandFailedError
|
2015-03-26 17:52:10 +00:00
|
|
|
from tasks.cephfs.cephfs_test_case import CephFSTestCase
|
2014-11-20 06:55:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
# Arbitrary timeouts for operations involving restarting
|
|
|
|
# an MDS or waiting for it to come up
|
|
|
|
MDS_RESTART_GRACE = 60
|
|
|
|
|
|
|
|
|
2014-12-18 13:03:40 +00:00
|
|
|
class TestMDSAutoRepair(CephFSTestCase):
|
2014-11-20 06:55:39 +00:00
|
|
|
def test_backtrace_repair(self):
|
|
|
|
"""
|
|
|
|
MDS should verify/fix backtrace on fetch dirfrag
|
|
|
|
"""
|
|
|
|
|
2015-11-29 11:48:29 +00:00
|
|
|
self.mount_a.run_shell(["mkdir", "testdir1"])
|
|
|
|
self.mount_a.run_shell(["touch", "testdir1/testfile"])
|
2015-11-02 17:01:00 +00:00
|
|
|
dir_objname = "{:x}.00000000".format(self.mount_a.path_to_ino("testdir1"))
|
2014-11-20 06:55:39 +00:00
|
|
|
|
2015-06-29 09:37:23 +00:00
|
|
|
# drop inodes caps
|
|
|
|
self.mount_a.umount_wait()
|
|
|
|
|
2015-11-02 17:01:00 +00:00
|
|
|
# flush journal entries to dirfrag objects, and expire journal
|
2014-11-20 06:55:39 +00:00
|
|
|
self.fs.mds_asok(['flush', 'journal'])
|
|
|
|
|
2015-11-02 17:01:00 +00:00
|
|
|
# Restart the MDS to drop the metadata cache (because we expired the journal,
|
|
|
|
# nothing gets replayed into cache on restart)
|
2021-03-11 19:06:23 +00:00
|
|
|
self.fs.rank_fail()
|
2015-11-02 17:01:00 +00:00
|
|
|
self.fs.wait_for_daemons()
|
2014-11-20 06:55:39 +00:00
|
|
|
|
|
|
|
# remove testdir1's backtrace
|
2021-03-10 22:46:27 +00:00
|
|
|
self.fs.radosm(["rmxattr", dir_objname, "parent"])
|
2014-11-20 06:55:39 +00:00
|
|
|
|
|
|
|
# readdir (fetch dirfrag) should fix testdir1's backtrace
|
2020-04-03 09:26:22 +00:00
|
|
|
self.mount_a.mount_wait()
|
2015-11-29 11:48:29 +00:00
|
|
|
self.mount_a.run_shell(["ls", "testdir1"])
|
2014-11-20 06:55:39 +00:00
|
|
|
|
|
|
|
# flush journal entries to dirfrag objects
|
|
|
|
self.fs.mds_asok(['flush', 'journal'])
|
|
|
|
|
|
|
|
# check if backtrace exists
|
2021-03-10 22:46:27 +00:00
|
|
|
self.fs.radosm(["getxattr", dir_objname, "parent"])
|
2014-11-20 06:55:39 +00:00
|
|
|
|
2014-12-05 14:01:13 +00:00
|
|
|
def test_mds_readonly(self):
|
|
|
|
"""
|
|
|
|
test if MDS behave correct when it's readonly
|
|
|
|
"""
|
|
|
|
# operation should successd when MDS is not readonly
|
2015-11-29 11:48:29 +00:00
|
|
|
self.mount_a.run_shell(["touch", "test_file1"])
|
2014-12-05 14:01:13 +00:00
|
|
|
writer = self.mount_a.write_background(loop=True)
|
|
|
|
|
|
|
|
time.sleep(10)
|
|
|
|
self.assertFalse(writer.finished)
|
|
|
|
|
|
|
|
# force MDS to read-only mode
|
|
|
|
self.fs.mds_asok(['force_readonly'])
|
|
|
|
time.sleep(10)
|
|
|
|
|
|
|
|
# touching test file should fail
|
|
|
|
try:
|
2015-11-29 11:48:29 +00:00
|
|
|
self.mount_a.run_shell(["touch", "test_file1"])
|
2014-12-18 13:03:40 +00:00
|
|
|
except CommandFailedError:
|
2014-12-05 14:01:13 +00:00
|
|
|
pass
|
|
|
|
else:
|
|
|
|
self.assertTrue(False)
|
|
|
|
|
|
|
|
# background writer also should fail
|
|
|
|
self.assertTrue(writer.finished)
|
|
|
|
|
2016-04-06 16:17:32 +00:00
|
|
|
# The MDS should report its readonly health state to the mon
|
2017-07-11 03:39:31 +00:00
|
|
|
self.wait_for_health("MDS_READ_ONLY", timeout=30)
|
2016-04-06 16:17:32 +00:00
|
|
|
|
2014-12-05 14:01:13 +00:00
|
|
|
# restart mds to make it writable
|
2015-02-07 09:33:58 +00:00
|
|
|
self.fs.mds_fail_restart()
|
2014-12-05 14:01:13 +00:00
|
|
|
self.fs.wait_for_daemons()
|
2016-04-06 16:17:32 +00:00
|
|
|
|
|
|
|
self.wait_for_health_clear(timeout=30)
|