1235 lines
41 KiB
Python
Executable File
1235 lines
41 KiB
Python
Executable File
#!/usr/bin/python
|
|
|
|
# Copyright (c) 2017 Red Hat, Inc. All rights reserved. This copyrighted material
|
|
# is made available to anyone wishing to use, modify, copy, or
|
|
# redistribute it subject to the terms and conditions of the GNU General
|
|
# Public License v.2.
|
|
#
|
|
# This program is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
|
|
# PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; If not, see http://www.gnu.org/licenses/.
|
|
#
|
|
# Author: Jakub Krysl <jkrysl@redhat.com>
|
|
|
|
import sys, os
|
|
sys.path.append(os.path.abspath("dmpd_library.py"))
|
|
|
|
from dmpd_library import *
|
|
|
|
def thin_init(args):
|
|
# Create thin pool with LVs
|
|
print("INFO: Initializing test case")
|
|
errors = []
|
|
|
|
atomic_run("Creating loopdev",
|
|
name=args["loop1"],
|
|
size=args["loop1_size"],
|
|
command=loopdev.create_loopdev,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating VG",
|
|
vg_name=args["group"],
|
|
pv_name="/dev/" + args["loop1"],
|
|
command=lvm.vg_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating thin pool",
|
|
vg_name=args["group"],
|
|
lv_name=args["pool"],
|
|
options=["-T", "-L 500"],
|
|
command=lvm.lv_create,
|
|
errors=errors)
|
|
|
|
# create few LVs to increase transaction ID and be able to do thin_delta
|
|
for i in range(args["number of vols"]):
|
|
atomic_run("Creating thin LV No. %s" % i,
|
|
vg_name=args["group"] + "/" + args["pool"],
|
|
lv_name=args["vol"] + str(i),
|
|
options=["-T", "-V 100"],
|
|
command=lvm.lv_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating filesystem on LV No. %s" % i,
|
|
vg_name=args["group"],
|
|
lv_name=args["vol"] + str(i),
|
|
command=create_filesystem,
|
|
errors=errors)
|
|
|
|
atomic_run("Deactivating thin LV No. %s" % i,
|
|
lv_name=args["vol"] + str(i),
|
|
vg_name=args["group"],
|
|
command=lvm.lv_deactivate,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating metadata snapshot",
|
|
lv_name=args["pool"],
|
|
vg_name=args["group"],
|
|
command=metadata_snapshot,
|
|
errors=errors)
|
|
|
|
atomic_run("Deactivating pool",
|
|
lv_name=args["pool"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_deactivate,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating swap LV",
|
|
vg_name=args["group"],
|
|
lv_name=args["swap"],
|
|
options=["-L 100"],
|
|
command=lvm.lv_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Deactivating swap",
|
|
lv_name=args["swap"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_deactivate,
|
|
errors=errors)
|
|
|
|
atomic_run("Swapping metadata",
|
|
vg_name=args["group"],
|
|
lv_name=args["swap"],
|
|
options=["-y", "--thinpool " + args["group"] + "/" + args["pool"],
|
|
"--poolmetadata "],
|
|
command=lvm.lv_convert,
|
|
errors=errors)
|
|
|
|
atomic_run("Activating swap",
|
|
lv_name=args["swap"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_activate,
|
|
errors=errors)
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Initialization passed")
|
|
else:
|
|
TC.tfail("Initialization failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def thin_clean(args):
|
|
print("INFO: Cleaning up")
|
|
errors = []
|
|
|
|
# restoring metadata device in case it is corrupted
|
|
atomic_run("Repairing metadata device",
|
|
source_file="/tmp/metadata",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
quiet=True,
|
|
command=dmpd.thin_restore,
|
|
errors=errors)
|
|
|
|
# thinpool got activated after checking its metadata to get bad checksum
|
|
atomic_run("Deactivating pool",
|
|
lv_name=args["pool"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_deactivate,
|
|
errors=errors)
|
|
|
|
atomic_run("Deactivating swap",
|
|
lv_name=args["swap"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_deactivate,
|
|
errors=errors)
|
|
|
|
atomic_run("Swapping back metadata",
|
|
vg_name=args["group"],
|
|
lv_name=args["swap"],
|
|
options=["-y", "--thinpool " + args["group"] + "/" + args["pool"],
|
|
"--poolmetadata "],
|
|
command=lvm.lv_convert,
|
|
errors=errors)
|
|
|
|
atomic_run("Removing swap",
|
|
lv_name=args["swap"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_remove,
|
|
errors=errors)
|
|
|
|
atomic_run("Removing thinpool",
|
|
lv_name=args["pool"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_remove,
|
|
errors=errors)
|
|
|
|
atomic_run("Removing VG",
|
|
vg_name=args["group"],
|
|
force=True,
|
|
command=lvm.vg_remove,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting loopdev",
|
|
name=args["loop1"],
|
|
command=loopdev.delete_loopdev,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting metadata file",
|
|
cmd="rm -f /tmp/metadata",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting repair metadata file",
|
|
cmd="rm -f /tmp/metadata_repair",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting snapshot metadata file",
|
|
cmd="rm -f /tmp/metadata_snap",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Cleanup passed")
|
|
else:
|
|
TC.tfail("Cleanup failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
print(errors)
|
|
return 1
|
|
return 0
|
|
|
|
def thin_test(args):
|
|
print("\n#######################################\n")
|
|
print(
|
|
"INFO: Testing thin tools runtime provided by device_mapper_persistent_data")
|
|
|
|
errors = []
|
|
|
|
atomic_run("Checking metadata",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.thin_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata with few paramethers",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
super_block_only=True,
|
|
skip_mappings=True,
|
|
ignore_non_fatal_errors=True,
|
|
command=dmpd.thin_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Listing information about thin LVs",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.thin_ls,
|
|
errors=errors)
|
|
|
|
atomic_run("Listing information about thin LVs without headers",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
no_headers=True,
|
|
command=dmpd.thin_ls,
|
|
errors=errors)
|
|
|
|
# Not yet in Fedora 26, shoud be in F27
|
|
#atomic_run("Dumping metadata to standard output without mappings",
|
|
# formatting="human_readable",
|
|
# source_vg=args["group"],
|
|
# source_lv=args["swap"],
|
|
# skip_mappings=True,
|
|
# command=dmpd.thin_dump,
|
|
# errors=errors)
|
|
|
|
atomic_run("Dumping metadata to standard output",
|
|
formatting="human_readable",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.thin_dump,
|
|
errors=errors)
|
|
|
|
atomic_run("Dumping metadata to standard output from snapshot",
|
|
formatting="human_readable",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
snapshot=True,
|
|
command=dmpd.thin_dump,
|
|
errors=errors)
|
|
|
|
# Not yet in Fedora 26, shoud be in F27
|
|
#atomic_run("Dumping metadata with dev-id",
|
|
# formatting="human_readable",
|
|
# source_vg=args["group"],
|
|
# source_lv=args["swap"],
|
|
# dev_id=args["number of vols"] - 1,
|
|
# command=dmpd.thin_dump,
|
|
# errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for pool of 64k blocks and 100M size",
|
|
cmd="thin_metadata_size -b64k -s100m -m1 -um",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for pool of 64k blocks and 100M size",
|
|
cmd="thin_metadata_size -b64k -s100m -m1 -um -n",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for pool of 64k blocks and 100M size",
|
|
cmd="thin_metadata_size -b64k -s100m -m1 -um -nlong",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for pool of 64k blocks and 100M size",
|
|
cmd="thin_metadata_size -b64k -s100m -m1 -um -nshort",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Outputting reverse map of metadata device",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
region="0..-1",
|
|
command=dmpd.thin_rmap,
|
|
errors=errors)
|
|
|
|
# this fails now and it should not
|
|
# atomic_run("Discarding free space of pool",
|
|
# target_vg=args["group"],
|
|
# target_lv=args["swap"],
|
|
# command=dmpd.thin_trim,
|
|
# errors=errors)
|
|
|
|
atomic_run("Dumping metadata to file",
|
|
formatting="xml",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
repair=True,
|
|
output="/tmp/metadata",
|
|
command=dmpd.thin_dump,
|
|
errors=errors)
|
|
|
|
atomic_run("Dumping metadata to file from snapshot",
|
|
formatting="xml",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
snapshot=True,
|
|
output="/tmp/metadata_snap",
|
|
command=dmpd.thin_dump,
|
|
errors=errors)
|
|
|
|
atomic_run("Getting differences between thin LVs",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
thin1=1,
|
|
thin2=args["number of vols"] - 1,
|
|
snapshot=True,
|
|
command=dmpd.thin_delta,
|
|
errors=errors)
|
|
|
|
atomic_run("Getting differences between thin LVs with --verbose",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
thin1=1,
|
|
thin2=args["number of vols"] - 1,
|
|
verbosity=True,
|
|
snapshot=True,
|
|
command=dmpd.thin_delta,
|
|
errors=errors)
|
|
|
|
atomic_run("Getting differences between the same LV",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
thin1=1,
|
|
thin2=1,
|
|
snapshot=True,
|
|
command=dmpd.thin_delta,
|
|
errors=errors)
|
|
|
|
atomic_run("Getting differences between the same LV with --verbose",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
thin1=1,
|
|
thin2=1,
|
|
verbosity=True,
|
|
snapshot=True,
|
|
command=dmpd.thin_delta,
|
|
errors=errors)
|
|
|
|
atomic_run("Listing metadata output from snapshot",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
snapshot=True,
|
|
command=dmpd.thin_ls,
|
|
errors=errors)
|
|
|
|
# Need to run everything on snapshot before this as thin_restore removes the metadata snapshot
|
|
# This should work but is not working due to a bug
|
|
#atomic_run("Restoring metadata",
|
|
# source_file="/tmp/metadata_snap",
|
|
# target_vg=args["group"],
|
|
# target_lv=args["swap"],
|
|
# command=dmpd.thin_restore,
|
|
# errors=errors)
|
|
|
|
atomic_run("Restoring metadata",
|
|
source_file="/tmp/metadata",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
command=dmpd.thin_restore,
|
|
errors=errors)
|
|
|
|
# Repairing from non-binary file leads to segmentation fault
|
|
#atomic_run("Repairing metadata from file",
|
|
# source_file="/tmp/metadata",
|
|
# target_vg=args["group"],
|
|
# target_lv=args["swap"],
|
|
# command=dmpd.thin_repair,
|
|
# errors=errors)
|
|
|
|
atomic_run("Repairing metadata to file",
|
|
target_file="/tmp/metadata_repair",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.thin_repair,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata from file",
|
|
source_file="/tmp/metadata_repair",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
command=dmpd.thin_repair,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.thin_check,
|
|
errors=errors)
|
|
|
|
print("\n#######################################\n")
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Testing thin tools of device_mapper_persistent_data passed")
|
|
else:
|
|
TC.tfail("Testing thin tools of device_mapper_persistent_data failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def thin_errors_test(args):
|
|
print("\n#######################################\n")
|
|
print(
|
|
"INFO: Testing thin tools errors provided by device_mapper_persistent_data")
|
|
|
|
errors = []
|
|
|
|
# "thin_show_duplicates" does not work yet
|
|
functions = ["thin_check", "thin_delta", "thin_dump", "thin_ls", "thin_metadata_size", "thin_repair",
|
|
"thin_restore", "thin_rmap", "thin_trim"]
|
|
|
|
# Sanity to check for missing input
|
|
for func in functions:
|
|
atomic_run("Validating missing input",
|
|
False,
|
|
cmd=func,
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Sanity to check with wrong input
|
|
for func in functions:
|
|
atomic_run("Validating wrong input",
|
|
False,
|
|
cmd=func + " wrong",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Sanity to check with wrong option
|
|
for func in functions:
|
|
atomic_run("Validating wrong option",
|
|
False,
|
|
cmd=func + " -wrong",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Sanity to check present functions with -h
|
|
for func in functions:
|
|
atomic_run("Checking help of command",
|
|
cmd=func,
|
|
command=dmpd.get_help,
|
|
errors=errors)
|
|
|
|
# Sanity to check present functions with -V
|
|
for func in functions:
|
|
# thin_metadata_size has wrong return value
|
|
if func == "thin_metadata_size":
|
|
continue
|
|
atomic_run("Checking version of command",
|
|
cmd=func,
|
|
command=dmpd.get_version,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking original pool metadata, should fail",
|
|
False,
|
|
source_vg=args["group"],
|
|
source_lv=args["pool"],
|
|
command=dmpd.thin_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Listing information about thin LVs",
|
|
False,
|
|
cmd="thin_ls /dev/mapper/%s-%s --format \"WRONG\"" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -b 64",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -b 64 -s 128",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -b 25 -s 128 -m 10",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -b 128 -s 64 -m 10",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -u h",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -n -n",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -nlongshort",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking thin_metadata_size inputs",
|
|
False,
|
|
cmd="thin_metadata_size -b 128 -b 64",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata without output",
|
|
False,
|
|
cmd="thin_repair -i /tmp/metadata_repair",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Dumping metadata with wrong custom format",
|
|
False,
|
|
cmd="thin_dump /dev/mapper/%s-%s --format custom=wrong" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Dumping metadata with unknown format",
|
|
False,
|
|
cmd="thin_dump /dev/mapper/%s-%s --format wrong" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Dumping metadata with wrong dev-id",
|
|
False,
|
|
cmd="thin_dump /dev/mapper/%s-%s --dev-id wrong" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata to produce 'output file does not exist' error",
|
|
False,
|
|
cmd="thin_repair -i /dev/mapper/%s-%s -o /tmp/wrong.wrong" %
|
|
(args['group'], args['swap']),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata to produce 'output file too small' error",
|
|
False,
|
|
cmd="thin_repair -i /tmp/metadata -o /tmp/metadata",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# This does not fail now due to a bug
|
|
#atomic_run("Outputting reverse map of metadata device, should fail without region",
|
|
# cmd="thin_rmap /dev/mapper/%s-%s" % (args["group"], args["swap"]),
|
|
# command=run,
|
|
# errors=errors)
|
|
|
|
atomic_run("Outputting reverse map of metadata device with wrong region",
|
|
False,
|
|
cmd="thin_rmap /dev/mapper/%s-%s --region 0..0" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Outputting reverse map of metadata device with wrong region",
|
|
False,
|
|
cmd="thin_rmap /dev/mapper/%s-%s --region 0...1" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Outputting reverse map of metadata device with wrong region",
|
|
False,
|
|
cmd="thin_rmap /dev/mapper/%s-%s --region 00" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Outputting reverse map of metadata device with wrong device",
|
|
False,
|
|
cmd="thin_rmap --region 0..-1 /tmp/wrong.wrong",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Reverse mapping from bad file leads to segmentation fault
|
|
#atomic_run("Outputting reverse map of metadata device with wrong device",
|
|
# False,
|
|
# cmd="thin_rmap --region 0..-1 /tmp/metadata",
|
|
# command=run,
|
|
# errors=errors)
|
|
|
|
atomic_run("Getting differences with thin1 ID out of range",
|
|
False,
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
thin1=-1,
|
|
thin2=args["number of vols"] - 1,
|
|
command=dmpd.thin_delta,
|
|
errors=errors)
|
|
|
|
atomic_run("Getting differences with thin2 ID out of range",
|
|
False,
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
thin1=1,
|
|
thin2=args["number of vols"] + 1,
|
|
command=dmpd.thin_delta,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata without output",
|
|
False,
|
|
cmd="thin_restore -i /tmp/metadata",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata with wrong options",
|
|
False,
|
|
cmd="thin_restore -i /tmp/metadata -o /dev/mapper/%s-%s --wrong test" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata with wrong source",
|
|
False,
|
|
cmd="thin_restore -i /tmp/wrong.wrong -o /dev/mapper/%s-%s" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Getting differences without thin2",
|
|
False,
|
|
cmd="thin_delta --thin1 1 /dev/mapper/%s-%s" %
|
|
(args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Corrupting metadata on device",
|
|
cmd="echo 'nothing' >> /dev/mapper/%s-%s" %
|
|
(args['group'], args['swap']),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Trying to fail while repairing metadata",
|
|
False,
|
|
source_vg=args['group'],
|
|
source_lv=args['swap'],
|
|
target_file="/tmp/metadata_repair",
|
|
command=dmpd.thin_repair,
|
|
errors=errors)
|
|
|
|
atomic_run("Trying to fail listing volumes",
|
|
False,
|
|
source_vg=args['group'],
|
|
source_lv=args['swap'],
|
|
command=dmpd.thin_ls,
|
|
errors=errors)
|
|
|
|
atomic_run("Trying to fail while checking metadata",
|
|
False,
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.thin_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Trying to fail while dumping metadata from snapshot",
|
|
False,
|
|
formatting="human_readable",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
snapshot=True,
|
|
command=dmpd.thin_dump,
|
|
errors=errors)
|
|
|
|
# restoring metadata device after corrupting it
|
|
atomic_run("Repairing metadata device",
|
|
source_file="/tmp/metadata",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
quiet=True,
|
|
command=dmpd.thin_restore,
|
|
errors=errors)
|
|
|
|
print("\n#######################################\n")
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Testing thin tools errors of device_mapper_persistent_data passed")
|
|
else:
|
|
TC.tfail("Testing thin tools errors of device_mapper_persistent_data failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def cache_init(args):
|
|
print("INFO: Initializing test case")
|
|
errors = []
|
|
|
|
atomic_run("Creating loopdev 1 - 'fast' device",
|
|
name=args["loop1"],
|
|
size=args["loop1_size"],
|
|
command=loopdev.create_loopdev,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating loopdev 2 - 'slow' device",
|
|
name=args["loop2"],
|
|
size=args["loop2_size"],
|
|
command=loopdev.create_loopdev,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating VG",
|
|
vg_name=args["group"],
|
|
pv_name="/dev/" + args["loop1"] +
|
|
" /dev/" + args["loop2"],
|
|
command=lvm.vg_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating cache metadata volume",
|
|
vg_name=args["group"] + " /dev/" + args["loop1"],
|
|
lv_name=args["meta"],
|
|
options=["-L 12"],
|
|
command=lvm.lv_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating origin volume",
|
|
vg_name=args["group"] + " /dev/" + args["loop2"],
|
|
lv_name=args["origin"],
|
|
options=["-L 2G"],
|
|
command=lvm.lv_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating cache data volume",
|
|
vg_name=args["group"] + " /dev/" + args["loop1"],
|
|
lv_name=args["data"],
|
|
options=["-L 1G"],
|
|
command=lvm.lv_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating cache pool",
|
|
vg_name=args["group"],
|
|
lv_name=args["data"],
|
|
options=["-y --type cache-pool", "--cachemode writeback", "--poolmetadata %s/%s" %
|
|
(args["group"], args["meta"])],
|
|
command=lvm.lv_convert,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating cache logical volume",
|
|
vg_name=args["group"],
|
|
lv_name=args["origin"],
|
|
options=["-y", "--type cache", "--cachepool %s/%s" %
|
|
(args["group"], args["data"])],
|
|
command=lvm.lv_convert,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating filesystem on cache logical volume",
|
|
vg_name=args["group"],
|
|
lv_name=args["origin"],
|
|
command=create_filesystem,
|
|
errors=errors)
|
|
|
|
atomic_run("Splitting cache logical volume",
|
|
vg_name=args["group"],
|
|
lv_name=args["origin"],
|
|
options=["-y", "--splitcache"],
|
|
command=lvm.lv_convert,
|
|
errors=errors)
|
|
|
|
atomic_run("Creating swap LV",
|
|
vg_name=args["group"],
|
|
lv_name=args["swap"],
|
|
options=["-L 100"],
|
|
command=lvm.lv_create,
|
|
errors=errors)
|
|
|
|
atomic_run("Swapping metadata",
|
|
vg_name=args["group"],
|
|
lv_name=args["swap"],
|
|
options=["-y", "--cachepool " + args["group"] + "/" + args["data"],
|
|
"--poolmetadata "],
|
|
command=lvm.lv_convert,
|
|
errors=errors)
|
|
|
|
atomic_run("Activating swap",
|
|
lv_name=args["swap"],
|
|
vg_name=args["group"],
|
|
command=lvm.lv_activate,
|
|
errors=errors)
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Initialization passed")
|
|
else:
|
|
TC.tfail("Initialization failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def cache_clean(args):
|
|
print("INFO: Cleaning up")
|
|
errors = []
|
|
|
|
atomic_run("Removing VG",
|
|
vg_name=args["group"],
|
|
force=True,
|
|
command=lvm.vg_remove,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting loopdev loop1",
|
|
name=args["loop1"],
|
|
command=loopdev.delete_loopdev,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting loopdev loop2",
|
|
name=args["loop2"],
|
|
command=loopdev.delete_loopdev,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting metadata file",
|
|
cmd="rm -f /tmp/metadata",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Deleting repair metadata file",
|
|
cmd="rm -f /tmp/metadata_repair",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Cleanup passed")
|
|
else:
|
|
TC.tfail("Cleanup failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
print(errors)
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def cache_test(args):
|
|
print("\n#######################################\n")
|
|
print("INFO: Testing cache tools runtime provided by device_mapper_persistent_data")
|
|
|
|
errors = []
|
|
|
|
atomic_run("Checking metadata",
|
|
source_lv=args["swap"],
|
|
source_vg=args["group"],
|
|
command=dmpd.cache_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata with clear-need-check-flag",
|
|
source_lv=args["swap"],
|
|
source_vg=args["group"],
|
|
clear_needs_check_flag=True,
|
|
command=dmpd.cache_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata with super-block-only",
|
|
source_lv=args["swap"],
|
|
source_vg=args["group"],
|
|
super_block_only=True,
|
|
command=dmpd.cache_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata with few paramethers",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
skip_discards=True,
|
|
skip_mappings=True,
|
|
skip_hints=True,
|
|
command=dmpd.cache_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Dumping metadata to standard output",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.cache_dump,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for cache of 64 blocks and 128 size",
|
|
cmd="cache_metadata_size --block-size 64 --device-size 128",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for cache of 128 nr blocks",
|
|
cmd="cache_metadata_size --nr-blocks 128 --max-hint-width 4",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Dumping metadata to file",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
repair=True,
|
|
output="/tmp/metadata",
|
|
command=dmpd.cache_dump,
|
|
errors=errors)
|
|
|
|
# Not yet in Fedora 26, shoud be in F27
|
|
#atomic_run("Checking metadata file",
|
|
# source_file="/tmp/metadata",
|
|
# command=dmpd.cache_check,
|
|
# errors=errors)
|
|
#
|
|
#atomic_run("Restoring metadata with options",
|
|
# source_file="/tmp/metadata",
|
|
# target_vg=args["group"],
|
|
# target_lv=args["swap"],
|
|
# quiet=True,
|
|
# override_metadata_version=1,
|
|
# metadata_version=1,
|
|
# command=dmpd.cache_restore,
|
|
# errors=errors)
|
|
|
|
atomic_run("Restoring metadata from file",
|
|
source_file="/tmp/metadata",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
command=dmpd.cache_restore,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata to file",
|
|
target_file="/tmp/metadata_repair",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.cache_repair,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata from file",
|
|
source_file="/tmp/metadata_repair",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
command=dmpd.cache_repair,
|
|
errors=errors)
|
|
|
|
atomic_run("Simulating TTY for cache_restore",
|
|
cmd="script --return -c 'cache_restore -i /tmp/metadata -o /dev/mapper/%s-%s' /dev/null" %
|
|
(args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata",
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
quiet=True,
|
|
command=dmpd.cache_check,
|
|
errors=errors)
|
|
|
|
print("\n#######################################\n")
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Testing cache tools of device_mapper_persistent_data passed")
|
|
else:
|
|
TC.tfail("Testing cache tools of device_mapper_persistent_data failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def cache_errors_test(args):
|
|
print("\n#######################################\n")
|
|
print("INFO: Testing cache tools errors provided by device_mapper_persistent_data")
|
|
|
|
errors = []
|
|
|
|
functions = ["cache_check", "cache_dump", "cache_metadata_size", "cache_repair", "cache_restore"]
|
|
|
|
# Sanity to check for missing input
|
|
for func in functions:
|
|
atomic_run("Validating missing input",
|
|
False,
|
|
cmd=func,
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Sanity to check with wrong input
|
|
for func in functions:
|
|
atomic_run("Validating wrong input",
|
|
False,
|
|
cmd=func + " wrong",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Sanity to check with wrong option
|
|
for func in functions:
|
|
atomic_run("Validating wrong option",
|
|
False,
|
|
cmd=func + " -wrong",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Sanity to check with wrong -- option
|
|
for func in functions:
|
|
atomic_run("Validating wrong -- option",
|
|
False,
|
|
cmd=func + " --wrong",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# Sanity to check present functions with -h
|
|
for func in functions:
|
|
atomic_run("Checking help of command",
|
|
cmd="%s" % func,
|
|
command=dmpd.get_help,
|
|
errors=errors)
|
|
|
|
# Sanity to check present functions with -V
|
|
for func in functions:
|
|
atomic_run("Checking version of command",
|
|
cmd="%s" % func,
|
|
command=dmpd.get_version,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata of non-metadata file",
|
|
False,
|
|
cmd="cache_check README",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata of non-existent file",
|
|
False,
|
|
cmd="cache_check WRONG",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking metadata of non-regular file",
|
|
False,
|
|
cmd="cache_check /dev/mapper/control",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for cache of 64 blocks",
|
|
False,
|
|
cmd="cache_metadata_size --block-size 64",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for cache of 128 size",
|
|
False,
|
|
cmd="cache_metadata_size --device-size 128",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Calculating metadata size for cache of 64 blocks and 128 size and 128 nr blocks",
|
|
False,
|
|
cmd="cache_metadata_size --block-size 64 --device-size 128 --nr-blocks 128",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata without output",
|
|
False,
|
|
cmd="cache_repair -i /tmp/metadata_repair",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata with wrong options",
|
|
False,
|
|
cmd="cache_restore -i /tmp/metadata -o /dev/mapper/%s-%s --wrong test" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata with wrong metadata version",
|
|
False,
|
|
source_file="/tmp/metadata",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
metadata_version=12445,
|
|
command=dmpd.cache_restore,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata with wrong source",
|
|
False,
|
|
cmd="cache_restore -i /tmp/wrong.wrong -o /dev/mapper/%s-%s" % (args["group"], args["swap"]),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata with bit source",
|
|
False,
|
|
source_file="/tmp/metadata_repair",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
command=dmpd.cache_restore,
|
|
errors=errors)
|
|
|
|
atomic_run("Restoring metadata without output",
|
|
False,
|
|
cmd="cache_restore -i /tmp/metadata",
|
|
command=run,
|
|
errors=errors)
|
|
|
|
# I am not able to run cache_restore with --omit-clean-shutdown successfully
|
|
#atomic_run("Restoring metadata with options",
|
|
# source_file="/tmp/metadata",
|
|
# target_vg=args["group"],
|
|
# target_lv=args["swap"],
|
|
# omit_clean_shutdown=True,
|
|
# command=dmpd.cache_restore,
|
|
# errors=errors)
|
|
|
|
# This fails in Fedora 26, should work in F27
|
|
#atomic_run("Checking metadata",
|
|
# source_vg=args["group"],
|
|
# source_lv=args["swap"],
|
|
# command=dmpd.cache_check,
|
|
# errors=errors)
|
|
|
|
#FIXME: Find other way to corrupt metadata, this exploits a bug
|
|
atomic_run("Corrupting mappings on metadata device",
|
|
False,
|
|
source_file="Makefile",
|
|
target_vg=args["group"],
|
|
target_lv=args["swap"],
|
|
command=dmpd.cache_restore,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking corrupted mappings",
|
|
False,
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.cache_check,
|
|
errors=errors)
|
|
|
|
atomic_run("Trying to fail while dumping metadata",
|
|
False,
|
|
source_vg=args['group'],
|
|
source_lv=args['swap'],
|
|
output="/tmp/metadata",
|
|
command=dmpd.cache_dump,
|
|
errors=errors)
|
|
|
|
atomic_run("Repairing metadata",
|
|
source_vg=args['group'],
|
|
source_lv=args['swap'],
|
|
target_file="/tmp/metadata_repair",
|
|
command=dmpd.cache_repair,
|
|
errors=errors)
|
|
|
|
atomic_run("Corrupting metadata on device",
|
|
cmd="echo 'nothing' >> /dev/mapper/%s-%s" % (args['group'], args['swap']),
|
|
command=run,
|
|
errors=errors)
|
|
|
|
atomic_run("Trying to fail while repairing metadata",
|
|
False,
|
|
source_vg=args['group'],
|
|
source_lv=args['swap'],
|
|
target_file="/tmp/metadata_repair",
|
|
command=dmpd.cache_repair,
|
|
errors=errors)
|
|
|
|
atomic_run("Trying to fail while dumping metadata",
|
|
False,
|
|
source_vg=args['group'],
|
|
source_lv=args['swap'],
|
|
output="/tmp/metadata",
|
|
command=dmpd.cache_dump,
|
|
errors=errors)
|
|
|
|
atomic_run("Checking corrupted metadata",
|
|
False,
|
|
source_vg=args["group"],
|
|
source_lv=args["swap"],
|
|
command=dmpd.cache_check,
|
|
errors=errors)
|
|
|
|
|
|
print("\n#######################################\n")
|
|
|
|
if len(errors) == 0:
|
|
TC.tpass("Testing cache tools errors of device_mapper_persistent_data passed")
|
|
else:
|
|
TC.tfail("Testing cache tools errors of device_mapper_persistent_data failed with following errors: \n\t'" +
|
|
"\n\t ".join([str(i) for i in errors]))
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def main():
|
|
# Initialize Test Case
|
|
global TC
|
|
TC = TestClass()
|
|
|
|
# Initialize library classes
|
|
global loopdev
|
|
global lvm
|
|
global dmpd
|
|
loopdev = LoopDev()
|
|
lvm = LVM()
|
|
dmpd = DMPD()
|
|
|
|
args = {"loop1": "loop1",
|
|
"loop1_size": 2048,
|
|
"loop2": "loop2",
|
|
"loop2_size": 4128,
|
|
"group": "vgtest",
|
|
"origin": "origin",
|
|
"data": "cache_data",
|
|
"meta": "cache_meta",
|
|
"pool": "pool",
|
|
"vol": "thinvol",
|
|
"number of vols": 10,
|
|
"swap": "swapvol"}
|
|
|
|
# Initialization
|
|
install_package("device-mapper-persistent-data")
|
|
|
|
# Tests for thin tools provided by device-mapper-persistent-data
|
|
thin_init(args)
|
|
thin_test(args)
|
|
thin_errors_test(args)
|
|
thin_clean(args)
|
|
|
|
# Tests for cache tools provided by device-mapper-persistent-data
|
|
cache_init(args)
|
|
cache_test(args)
|
|
cache_errors_test(args)
|
|
cache_clean(args)
|
|
|
|
if not TC.tend():
|
|
print("FAIL: test failed")
|
|
sys.exit(1)
|
|
|
|
print("PASS: Test pass")
|
|
sys.exit(0)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|