ansitest/ZFS/zfs-drive-slicer.sh
2022-01-27 14:27:52 -06:00

160 lines
6.0 KiB
Bash

#!/bin/bash
# Get me a set of disks for DRAID
# DONE [b..y] a[a..x] get slices of X disks and be able to verify with wc -w
# REQUIRES: seq
# Trick to put header outside of col -t
>&2 echo "$0 - 2021 Dave Bechtel"
>&2 echo "Pass arg1=total disks in pool -- arg2=how many disks per vdev"
>&2 echo "+ NOTE arg2 ^^ should factor in the RAIDz level 1/2/3 desired to sustain X number"
>&2 echo "+ of failed disks per vdev + vspares, dont go too narrow or will lose capacity"
>&2 echo "NOTE it is Highly Recommended to export the pool after creation with shortnames"
>&2 echo "+ and re-import with -d /dev/disk/by-id or other long form names"
>&2 echo "NOTE output lines should be the same number of devices to balance"
>&2 echo "PROTIP: Pipe output to column -t to make it look nice"
>&2 echo "Omit the trailing slash on last line when copying to clipboard"
>&2 echo "================================================================================="
# REF: https://tldp.org/LDP/abs/html/arrays.html
# regular array - STARTS AT 0
declare -a fullset=(sd{b..y} sda{a..x} sdb{a..x} sdc{a..x}) # Total 96, sets of 24
# NOTE ^^ intentionally has gaps -- sdz, sday sdaz, sdby sdbz, sdcy sdcz == Reserved for spares (7)
# integer
declare -i howmany sliceby idx x
function slice () {
howmany=$1
sliceby=$2
idx=0
prefix="raidz2 "
[ "$sliceby" = "2" ] && prefix="mirror "
printf "${prefix} " # print initial
for x in $(seq 0 1 $howmany); do
[ $x -ge $howmany ] && break
let idx=$idx+1
# echo -n "${fullset[$idx]} "
printf "${fullset[$x]} " # no newline
if [ $idx -ge $sliceby ]; then
echo ' \'
printf "${prefix} "
idx=0
fi
done
echo ''
}
if [ "$1" = "" ]; then
# Demo
echo "o Copypasta each line not-including the '\' and verify with: echo '[paste]' |wc -w "
slice 72 72
echo '===== ^^ 72 / 72 -- One Big Mother vdev'
slice 72 36
echo '===== ^^ 72 / 36 -- 2 VDEVs'
slice 72 24
echo '===== ^^ 72 / 24 -- 3 VDEVs'
# = sd{b..y} sda{a..l} \
#sda{m..x} sdb{a..x}
exit; # early
fi
# Basic sanity
if [ "$1" -lt "$2" ]; then
echo "$0 - Failed sanity check, \$2 must be greater than \$1"
exit 999; # Somebody call Scotland Yard, we have a violation
fi
slice $1 $2
# This is a decent method because we can give it arbitrary numbers of disks (up to total defined)
# and divide as needed; try 26 2, 24 2, 32 4, 32 8
# NOTE all output lines should have the same length - if you dont you wont have a balanced set of disks
# e.g. 32 6 = invalid config (32 4 = valid) but we just give you output -- real sanity checks are up 2U
# HOW THIS WORKS:
# Imagine a set of disks sdb..sdcx , 96 in total, set along a slide rule
# "idx" is our slider/window
# If idx >= however many drives we need per vdev, it prints a "line continue" char and drops down a line -
# giving us the exact short drive names needed to create per-vdev
# Example for 60-bay Storinator from 45drives - gives us 56 usable disks in pool + 3 spares (sdz sday sdaz), and 4 vdevs
# $0 56 14 |column -t
#sdb sdc sdd sde sdf sdg sdh sdi sdj sdk sdl sdm sdn sdo \
#sdp sdq sdr sds sdt sdu sdv sdw sdx sdy sdaa sdab sdac sdad \
#sdae sdaf sdag sdah sdai sdaj sdak sdal sdam sdan sdao sdap sdaq sdar \
#sdas sdat sdau sdav sdaw sdax sdba sdbb sdbc sdbd sdbe sdbf sdbg sdbh \
# Example for 45drives with 1 spare (sdz), 4 vdevs of 11:
# $0 44 11 |column -t
#sdb sdc sdd sde sdf sdg sdh sdi sdj sdk sdl \
#sdm sdn sdo sdp sdq sdr sds sdt sdu sdv sdw \
#sdx sdy sdaa sdab sdac sdad sdae sdaf sdag sdah sdai \
#sdaj sdak sdal sdam sdan sdao sdap sdaq sdar sdas sdat \
#
# If you want more spares (5): (reserved sdz sdaq sdar sdas sdat) == 1 spare for every 8 drives
# $0 40 10 |column -t
#sdb sdc sdd sde sdf sdg sdh sdi sdj sdk \
#sdl sdm sdn sdo sdp sdq sdr sds sdt sdu \
#sdv sdw sdx sdy sdaa sdab sdac sdad sdae sdaf \
#sdag sdah sdai sdaj sdak sdal sdam sdan sdao sdap \
# HOWTO Verify output from this script
# $0 42 14 |column -t >/tmp/zfsds.txt
# sed -i 's/\\//g' /tmp/zfsds.txt # remove line-continuation chars
# cat /tmp/zfsds.txt
#sdb sdc sdd sde sdf sdg sdh sdi sdj sdk sdl sdm sdn sdo
#sdp sdq sdr sds sdt sdu sdv sdw sdx sdy sdaa sdab sdac sdad
#sdae sdaf sdag sdah sdai sdaj sdak sdal sdam sdan sdao sdap sdaq sdar
# while read line; do echo "$line" |wc -w; done < /tmp/zfsds.txt
# 14
# 14
# 14
# HOWTO get long-form disks:
# $0 arg1 arg2 >/tmp/zfsds.txt
# Then use drive-slicer-get-longform.sh
# ^ Requires output from this script
# 2022.0126 added default raidz2/mirror prefix; output is not perfect but does the job
# 2022.0127 LOL so basically I reinvented xargs
# Easier to work with sets of 24 to divide by 8
#(echo sd{b..y} sda{a..x} sdb{a..x} sdc{a..x} |xargs -n 8 echo raidz2 ) |column -t
#raidz2 sdb sdc sdd sde sdf sdg sdh sdi
#raidz2 sdj sdk sdl sdm sdn sdo sdp sdq
#raidz2 sdr sds sdt sdu sdv sdw sdx sdy
#raidz2 sdaa sdab sdac sdad sdae sdaf sdag sdah
#raidz2 sdai sdaj sdak sdal sdam sdan sdao sdap
#raidz2 sdaq sdar sdas sdat sdau sdav sdaw sdax
#raidz2 sdba sdbb sdbc sdbd sdbe sdbf sdbg sdbh
#raidz2 sdbi sdbj sdbk sdbl sdbm sdbn sdbo sdbp
#raidz2 sdbq sdbr sdbs sdbt sdbu sdbv sdbw sdbx
#raidz2 sdca sdcb sdcc sdcd sdce sdcf sdcg sdch
#raidz2 sdci sdcj sdck sdcl sdcm sdcn sdco sdcp
#raidz2 sdcq sdcr sdcs sdct sdcu sdcv sdcw sdcx
#echo sd{b..y} sda{a..x} sdb{a..x} sdc{a..x} |xargs -n 2 echo mirror |head -n 2
#mirror sdb sdc
#mirror sdd sde
# redirect output to >tmp.txt
# add a line-continue char at the end
# $ while read line; do printf "%s" "$line \\";echo ''; done < tmp.txt
#raidz2 sdb sdc sdd sde sdf sdg sdh sdi \
#raidz2 sdj sdk sdl sdm sdn sdo sdp sdq \
# Oh well, at least I add a line-continue and start/limit with a known number of drives :)