2022-01-27 16:23:59 -06:00
|
|
|
#!/bin/bash
|
|
|
|
|
|
|
|
#set -x
|
|
|
|
# PROTIP use bash5 from macports/brew on osx, much faster
|
|
|
|
# Get me a set of shortname disks for DRAID
|
|
|
|
# DONE [b..y] a[a..x] get slices of X disks and be able to verify with wc -w
|
|
|
|
# REQUIRES: xargs, cut
|
|
|
|
|
|
|
|
# Trick to put header outside of col -t
|
|
|
|
>&2 echo "$0 - 2022 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"
|
2022-01-27 18:37:48 -06:00
|
|
|
>&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 to avoid issues"
|
|
|
|
>&2 echo "NOTE output lines NEED to be the same number of devices to balance the pool!"
|
2022-01-27 16:23:59 -06:00
|
|
|
>&2 echo "DO NOT FORGET to prefix these lines with raidz2, mirror or whatever is applicable!"
|
|
|
|
>&2 echo "PROTIP: Pipe output to column -t to make it look nice"
|
|
|
|
>&2 echo "=================================================================================="
|
|
|
|
|
|
|
|
# REF: https://tldp.org/LDP/abs/html/arrays.html
|
|
|
|
# regular array - STARTS AT 0
|
|
|
|
# NOTE vv keep this commented, slows waaay down on bash 3.x osx
|
|
|
|
#declare -a fullset=(sd{b..y} sda{a..x} sdb{a..x} sdc{a..x}) sdd{a..x} # Total 120, sets of 24
|
|
|
|
# NOTE ^^ intentionally has gaps -- sdz, sday sdaz, sdby sdbz, sdcy sdcz, sddy sddz == Reserved for spares (9)
|
|
|
|
|
|
|
|
|
|
|
|
function slice () {
|
|
|
|
# (echo sd{b..y} sda{a..x} sdb{a..x} sdc{a..x} sdd{a..x}) |wc -w # to print sum-total number of drives
|
|
|
|
|
|
|
|
# cut prints X range of fields (limit number of total drives), xargs breaks them up by 2nd arg
|
|
|
|
(echo sd{b..y} sda{a..x} sdb{a..x} sdc{a..x} sdd{a..x}) \
|
|
|
|
|cut -d' ' -f1-$1 \
|
|
|
|
|xargs -n $2
|
|
|
|
}
|
|
|
|
|
|
|
|
if [ "$1" = "" ]; then
|
|
|
|
# Demo
|
|
|
|
echo "o Copypasta each line and verify with: echo '[paste]' |wc -w "
|
|
|
|
slice 72 72
|
|
|
|
echo '===== ^^ 72 / 72 -- One Big Mother vdev'
|
|
|
|
echo ''
|
|
|
|
slice 72 36
|
|
|
|
echo '===== ^^ 72 / 36 -- 2 VDEVs'
|
|
|
|
echo ''
|
|
|
|
slice 72 24
|
|
|
|
echo '===== ^^ 72 / 24 -- 3 VDEVs'
|
|
|
|
slice 96 12
|
|
|
|
echo '===== ^^ 96 / 12 -- 8 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
|
|
|
|
|
|
|
|
arg1=$1
|
|
|
|
arg2=$2
|
|
|
|
|
|
|
|
# REF: https://stackoverflow.com/questions/806906/how-do-i-test-if-a-variable-is-a-number-in-bash/806923
|
|
|
|
re='^[0-9]+$'
|
|
|
|
if ! [[ $arg1 =~ $re ]] ; then
|
|
|
|
echo "error: arg1 Not a number" >&2; exit 666
|
|
|
|
fi
|
|
|
|
if ! [[ $arg2 =~ $re ]] ; then
|
|
|
|
echo "error: arg1 Not a number" >&2; exit 666
|
|
|
|
fi
|
|
|
|
|
|
|
|
slice $arg1 $arg2
|
|
|
|
|
|
|
|
# DO NOT put exit in case we get SOURCEd for the function(?)
|
|
|
|
|
|
|
|
# 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
|
|
|
|
# "xargs" is our slider/window
|
|
|
|
# "cut" only prints up to the total number of drives we limit it to
|
|
|
|
# If $2 >= however many drives we need per vdev, it drops down a line -
|
|
|
|
# giving us the exact short drive names needed to create a zfs pool 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 \
|
|
|
|
|
|
|
|
# (echo sd{b..y} sda{a..x} sdb{a..x} sdc{a..x} sdd{a..x}) |cut -d' ' -f1-15 |xargs -n 5
|
|
|
|
#sdb sdc sdd sde sdf
|
|
|
|
#sdg sdh sdi sdj sdk
|
|
|
|
#sdl sdm sdn sdo sdp
|
|
|
|
|
|
|
|
|
|
|
|
# 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
|
|
|
|
|
|
|
|
# PROTIP to add in raidz2 prefix and trailing '\':
|
|
|
|
# ./zfs-drive-slicer-xargs-simplified.sh 80 8 |while read line; do printf "%s" "raidz2 $line \\";echo ''; done
|
|
|
|
|
|
|
|
# ./zfs-drive-slicer-xargs-simplified.sh 20 2 |while read line; do printf "%s" "mirror $line \\";echo ''; done
|
|
|
|
#mirror sdb sdc \
|
|
|
|
#mirror sdd sde \
|
|
|
|
#...
|
|
|
|
#mirror sdt sdu \
|
|
|
|
|
2022-01-27 18:37:48 -06:00
|
|
|
# Triple mirroring:
|
2022-01-27 16:23:59 -06:00
|
|
|
# $ ./zfs-drive-slicer-xargs-simplified.sh 15 3 |while read line; do printf "%s" "mirror $line \\";echo ''; done
|
|
|
|
#mirror sdb sdc sdd \
|
|
|
|
#mirror sde sdf sdg \
|
|
|
|
#...
|
|
|
|
#mirror sdn sdo sdp \
|
|
|
|
|
2022-01-27 18:37:48 -06:00
|
|
|
# should-be faster awk solution:
|
|
|
|
# ./zfs-drive-slicer-xargs-simplified.sh 90 10 |awk '{ print "raidz2 ", $0, "\\"}'
|
|
|
|
|
|
|
|
# same thing a different way:
|
|
|
|
# edit in place and subsitute the needed text before printing
|
|
|
|
# ./zfs-drive-slicer-xargs-simplified.sh 90 10 |awk '{ sub(/^/, "raidz2 "); sub(/$/, " \\"); print }'
|
|
|
|
# ^ regexp - for every line, at beginning of line, prefix with "raidz2 "
|
|
|
|
# $ regexp - find end of line and print literal backslash, finally display edited line
|
|
|
|
|
2022-01-27 16:23:59 -06:00
|
|
|
|
|
|
|
# 2022.0127 substantially refactored to use xargs and cut for simplicity
|
|
|
|
# works with bash 3.2.57 osx
|