2010-04-12 23:02:59 +04:00
#!/bin/bash
2012-03-16 16:59:43 +04:00
# Copyright (C) 2010-2012 Red Hat, Inc. All rights reserved.
2010-07-28 16:20:38 +04:00
#
# 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.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software Foundation,
# Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
2010-05-12 14:08:35 +04:00
# check.sh: assert various things about volumes
2012-03-16 16:59:43 +04:00
# USAGE:
2010-05-12 14:08:35 +04:00
# check linear VG LV
# check lv_on VG LV PV
# check mirror VG LV [LOGDEV|core]
# check mirror_nonredundant VG LV
# check mirror_legs VG LV N
# check mirror_images_on VG LV DEV [DEV...]
# ...
2012-03-16 16:59:43 +04:00
test -z " $BASH " || set -e -o pipefail
2011-01-05 03:16:18 +03:00
2012-03-16 16:59:43 +04:00
die( ) {
echo " $@ " >& 2
return 1
2011-01-05 03:16:18 +03:00
}
2010-05-12 09:55:08 +04:00
lvl( ) {
lvs -a --noheadings " $@ "
}
2010-04-12 23:02:59 +04:00
lvdevices( ) {
2012-03-16 16:59:43 +04:00
get lv_devices " $@ "
2010-04-12 23:02:59 +04:00
}
2012-03-16 16:59:43 +04:00
mirror_images_redundant( ) {
local vg = $1
local lv = $vg /$2
lvs -a $vg -o+devices
for i in $( lvdevices $lv ) ; do
echo " # $i : "
lvdevices $vg /$i | sort | uniq
done > check.tmp.all
2010-04-12 23:02:59 +04:00
2012-03-16 16:59:43 +04:00
( grep -v ^# check.tmp.all || true ) | sort | uniq -d > check.tmp
2010-04-12 23:02:59 +04:00
2012-03-16 16:59:43 +04:00
test $( cat check.tmp | wc -l) -eq 0 || \
die " mirror images of $lv expected redundant, but are not: " \
$( cat check.tmp.all)
}
2010-04-12 23:02:59 +04:00
2012-03-16 16:59:43 +04:00
lv_on( ) {
local lv = $1 /$2
( lvdevices $lv | grep -F " $3 " ) || \
die " LV $lv expected on $3 but is not: " \
$( lvdevices $lv )
test $( lvdevices $lv | grep -vF " $3 " | wc -l) -eq 0 || \
die " LV $lv contains unexpected devices: " \
$( lvdevices $lv )
2010-04-12 23:02:59 +04:00
}
2010-04-12 23:33:58 +04:00
mirror_images_on( ) {
2012-03-16 16:59:43 +04:00
local vg = $1
local lv = $2
2010-05-12 14:08:35 +04:00
shift 2
2012-03-16 16:59:43 +04:00
for i in $( lvdevices $lv ) ; do
2010-05-12 14:08:35 +04:00
lv_on $vg $lv $1
2010-04-12 23:33:58 +04:00
shift
done
}
2012-03-16 16:59:43 +04:00
mirror_log_on( ) {
local vg = $1
local lv = $2
local where = $3
2010-05-12 14:08:35 +04:00
if test " $where " = "core" ; then
2012-03-16 16:59:43 +04:00
get lv_field $vg /$lv mirror_log | not grep mlog
2010-05-12 14:08:35 +04:00
else
2012-03-16 16:59:43 +04:00
lv_on $vg ${ lv } _mlog " $where "
2010-05-12 14:08:35 +04:00
fi
2010-04-12 23:02:59 +04:00
}
2012-03-16 16:59:43 +04:00
lv_is_contiguous( ) {
local lv = $1 /$2
test $( lvl --segments $lv | wc -l) -eq 1 || \
die " LV $lv expected to be contiguous, but is not: " \
$( lvl --segments $lv )
2010-04-12 23:02:59 +04:00
}
2012-03-16 16:59:43 +04:00
lv_is_clung( ) {
local lv = $1 /$2
test $( lvdevices $lv | sort | uniq | wc -l) -eq 1 || \
die " LV $lv expected to be clung, but is not: " \
$( lvdevices $lv | sort | uniq)
2010-04-12 23:02:59 +04:00
}
2012-03-16 16:59:43 +04:00
mirror_images_contiguous( ) {
for i in $( lvdevices $1 /$2 ) ; do
lv_is_contiguous $1 $i
2010-04-12 23:02:59 +04:00
done
}
2012-03-16 16:59:43 +04:00
mirror_images_clung( ) {
for i in $( lvdevices $1 /$2 ) ; do
lv_is_clung $1 $i
2010-04-12 23:02:59 +04:00
done
}
mirror( ) {
2010-05-12 14:08:35 +04:00
mirror_nonredundant " $@ "
2012-03-16 16:59:43 +04:00
mirror_images_redundant $1 $2
2010-05-12 14:08:35 +04:00
}
mirror_nonredundant( ) {
2012-03-16 16:59:43 +04:00
local lv = $1 /$2
local attr = $( get lv_field $lv attr)
( echo " $attr " | grep " ^m....... $" >/dev/null) || {
if ( echo " $attr " | grep " ^o....... $" >/dev/null) &&
2011-01-28 19:08:39 +03:00
lvs -a | fgrep " [ ${ 2 } _mimage " >/dev/null; then
2011-01-07 16:03:46 +03:00
echo " TEST WARNING: $lv is a snapshot origin and looks like a mirror, "
echo "assuming it is actually a mirror"
else
2012-03-16 16:59:43 +04:00
die " $lv expected a mirror, but is not: " \
$( lvs $lv )
2011-01-07 16:03:46 +03:00
fi
2010-04-12 23:02:59 +04:00
}
2012-03-16 16:59:43 +04:00
test -z " $3 " || mirror_log_on $1 $2 " $3 "
2010-04-12 23:02:59 +04:00
}
2010-05-06 23:01:26 +04:00
mirror_legs( ) {
2012-03-16 16:59:43 +04:00
local expect = $3
test " $expect " -eq $( lvdevices $1 /$2 | wc -w)
2010-05-06 23:01:26 +04:00
}
2012-03-16 16:59:43 +04:00
mirror_no_temporaries( ) {
local vg = $1
local lv = $2
( lvl -o name $vg | grep $lv | not grep "tmp" ) || \
die " $lv has temporary mirror images unexpectedly: " \
$( lvl $vg | grep $lv )
2010-05-12 14:08:35 +04:00
}
2010-04-12 23:02:59 +04:00
linear( ) {
2012-03-16 16:59:43 +04:00
local lv = $1 /$2
test $( get lv_field $lv stripes -a) -eq 1 || \
die " $lv expected linear, but is not: " \
$( lvl $lv -o+devices)
2010-05-12 09:55:08 +04:00
}
2012-07-24 23:17:54 +04:00
# in_sync <VG> <LV>
# Works for "mirror" and "raid*"
in_sync( ) {
local a
local b
local idx
local type
local lvm_name = " $1 / $2 "
local dm_name = $( echo $lvm_name | sed s:-:--: | sed s:/:-:)
if ! a = ( ` dmsetup status $dm_name ` ) ; then
die " Unable to get sync status of $1 "
elif [ ${ a [2] } = "snapshot-origin" ] ; then
if ! a = ( ` dmsetup status ${ dm_name } -real` ) ; then
die " Unable to get sync status of $1 "
fi
fi
if [ ${ a [2] } = "raid" ] ; then
# Last argument is the sync ratio for RAID
idx = $(( ${# a [@] } - 1 ))
type = ${ a [3] }
elif [ ${ a [2] } = "mirror" ] ; then
# 4th Arg tells us how far to the sync ratio
idx = $(( ${ a [3] } + 4 ))
type = ${ a [2] }
else
die " Unable to get sync ratio for target type ' ${ a [2] } ' "
fi
b = ( $( echo ${ a [ $idx ] } | sed s:/:' ' :) )
if [ ${ b [0] } != ${ b [1] } ] ; then
echo " $lvm_name ( $type ) is not in-sync "
return 1
fi
if [ [ ${ a [ $(( $idx - 1 )) ] } = ~ a ] ] ; then
die " $lvm_name in-sync, but 'a' characters in health status "
fi
echo " $lvm_name ( $type ) is in-sync "
return 0
}
2010-05-12 09:55:08 +04:00
active( ) {
2012-03-16 16:59:43 +04:00
local lv = $1 /$2
2012-03-28 15:10:08 +04:00
( get lv_field $lv attr | grep " ^....a... $" >/dev/null) || \
2012-03-16 16:59:43 +04:00
die " $lv expected active, but lvs says it's not: " \
$( lvl $lv -o+devices)
2012-03-28 15:10:08 +04:00
dmsetup info $1 -$2 >/dev/null ||
die " $lv expected active, lvs thinks it is but there are no mappings! "
2010-05-12 09:55:08 +04:00
}
inactive( ) {
2012-03-16 16:59:43 +04:00
local lv = $1 /$2
( get lv_field $lv attr | grep " ^....[-isd]... $" >/dev/null) || \
die " $lv expected inactive, but lvs says it's not: " \
$( lvl $lv -o+devices)
2012-03-28 15:10:08 +04:00
not dmsetup info $1 -$2 2>/dev/null || \
die " $lv expected inactive, lvs thinks it is but there are mappings! "
2010-04-12 23:02:59 +04:00
}
2012-03-16 16:59:43 +04:00
# Check for list of LVs from given VG
2011-04-12 16:39:24 +04:00
lv_exists( ) {
2012-03-16 16:59:43 +04:00
local vg = $1
local lv =
while [ $# -gt 1 ] ; do
shift
lv = " $lv $vg / $1 "
done
lvl $lv & >/dev/null || \
die " $lv expected to exist but does not "
2011-04-12 16:39:24 +04:00
}
2012-03-16 16:59:43 +04:00
pv_field( ) {
local actual = $( get pv_field " $1 " " $2 " " ${ @ : 4 } " )
test " $actual " = " $3 " || \
die " pv_field: PV=\" $1 \", field=\" $2 \", actual=\" $actual \", expected=\" $3 \" "
2011-01-05 03:16:18 +03:00
}
2012-03-16 16:59:43 +04:00
vg_field( ) {
local actual = $( get vg_field $1 " $2 " " ${ @ : 4 } " )
test " $actual " = " $3 " || \
die " vg_field: vg= $1 , field=\" $2 \", actual=\" $actual \", expected=\" $3 \" "
2011-01-05 03:16:18 +03:00
}
2012-03-16 16:59:43 +04:00
lv_field( ) {
local actual = $( get lv_field $1 " $2 " " ${ @ : 4 } " )
test " $actual " = " $3 " || \
die " lv_field: lv= $lv , field=\" $2 \", actual=\" $actual \", expected=\" $3 \" "
2011-01-05 03:16:18 +03:00
}
2012-03-16 16:59:43 +04:00
compare_fields( ) {
local cmd1 = $1
local obj1 = $2
local field1 = $3
local cmd2 = $4
local obj2 = $5
local field2 = $6
local val1 = $( $cmd1 --noheadings -o " $field1 " " $obj1 " )
local val2 = $( $cmd2 --noheadings -o " $field2 " " $obj2 " )
test " $val1 " = " $val2 " || \
die " compare_fields $obj1 ( $field1 ): $val1 $obj2 ( $field2 ): $val2 "
2011-01-05 03:16:18 +03:00
}
2012-03-16 16:59:43 +04:00
compare_vg_field( ) {
local vg1 = $1
local vg2 = $2
local field = $3
local val1 = $( vgs --noheadings -o " $field " $vg1 )
local val2 = $( vgs --noheadings -o " $field " $vg2 )
test " $val1 " = " $val2 " || \
die " compare_vg_field: $vg1 : $val1 , $vg2 : $val2 "
2011-01-05 03:16:18 +03:00
}
2012-03-16 16:59:43 +04:00
pvlv_counts( ) {
2011-01-05 03:16:18 +03:00
local local_vg = $1
local num_pvs = $2
local num_lvs = $3
local num_snaps = $4
2012-03-16 16:59:43 +04:00
lvs -o+devices $local_vg
2011-01-05 03:16:18 +03:00
vg_field $local_vg pv_count $num_pvs
vg_field $local_vg lv_count $num_lvs
vg_field $local_vg snap_count $num_snaps
}
2012-03-16 16:59:43 +04:00
unset LVM_VALGRIND
2010-04-12 23:02:59 +04:00
" $@ "