X-Git-Url: https://iankelling.org/git/?p=distro-setup;a=blobdiff_plain;f=mount-latest-subvol;h=359b534337669090503de236c2ab30467c35b4dd;hp=e4c4d6b0a652b992f6feb7483cecd394bbf1040c;hb=0b6d44c7f3d567e0a26138509c8a24cb57c69b50;hpb=d46190aff6f5dc65bd39524e3937dc5765895b42 diff --git a/mount-latest-subvol b/mount-latest-subvol index e4c4d6b..359b534 100644 --- a/mount-latest-subvol +++ b/mount-latest-subvol @@ -14,220 +14,319 @@ # limitations under the License. # usage: mount-latest-subvol +# +# Note, at source location, intentionally not executable, run and read +# install-my-scripts. +cd / [[ $EUID == 0 ]] || exec sudo -E "$BASH_SOURCE" "$@" errcatch() { - set -E; shopt -s extdebug - _err-trap() { - err=$? - exec >&2 - set +x - echo "${BASH_SOURCE[1]}:${BASH_LINENO[0]}:in \`$BASH_COMMAND' returned $err" - bash-trace 2 - echo "$0: exiting with code $err" - exit $err - } - trap _err-trap ERR - set -o pipefail + set -E; shopt -s extdebug + _err-trap() { + err=$? + exec >&2 + set +x + echo "${BASH_SOURCE[1]}:${BASH_LINENO[0]}:in \`$BASH_COMMAND' returned $err" + bash-trace 2 + echo "$0: exiting with code $err" + exit $err + } + trap _err-trap ERR + set -o pipefail } bash-trace() { - local -i argc_index=0 frame i start=${1:-1} max_indent=8 indent - local source - local extdebug=false - if [[ $(shopt -p extdebug) == *-s* ]]; then - extdebug=true - fi + local -i argc_index=0 frame i start=${1:-1} max_indent=8 indent + local source + local extdebug=false + if [[ $(shopt -p extdebug) == *-s* ]]; then + extdebug=true + fi - for ((frame=0; frame < ${#FUNCNAME[@]}-1; frame++)); do - argc=${BASH_ARGC[frame]} - argc_index+=$argc - ((frame < start)) && continue - if (( ${#BASH_SOURCE[@]} > 1 )); then - source="${BASH_SOURCE[frame+1]}:${BASH_LINENO[frame]}:" - fi - indent=$((frame-start+1)) - indent=$((indent < max_indent ? indent : max_indent)) - printf "%${indent}s↳%sin \`%s" '' "$source" "${FUNCNAME[frame]}" - if $extdebug; then - for ((i=argc_index-1; i >= argc_index-argc; i--)); do - printf " %s" "${BASH_ARGV[i]}" - done - fi - echo \' - done + for ((frame=0; frame < ${#FUNCNAME[@]}-1; frame++)); do + argc=${BASH_ARGC[frame]} + argc_index+=$argc + ((frame < start)) && continue + if (( ${#BASH_SOURCE[@]} > 1 )); then + source="${BASH_SOURCE[frame+1]}:${BASH_LINENO[frame]}:" + fi + indent=$((frame-start+1)) + indent=$((indent < max_indent ? indent : max_indent)) + printf "%${indent}s↳%sin \`%s" '' "$source" "${FUNCNAME[frame]}" + if $extdebug; then + for ((i=argc_index-1; i >= argc_index-argc; i--)); do + printf " %s" "${BASH_ARGV[i]}" + done + fi + echo \' + done } errcatch tu() { - while read -r line; do - file="$1" - grep -xFq "$line" "$file" || tee -a "$file"<<<"$line" - done + while read -r line; do + file="$1" + grep -xFq "$line" "$file" || tee -a "$file"<<<"$line" + done } e() { printf "%s\n" "$*"; "$@"; } mnt() { - dir=$1 - if ! mountpoint $dir &>/dev/null; then - mkdir -p $dir - e mount $dir + dir=$1 + if ! mountpoint $dir &>/dev/null; then + mkdir -p $dir + e mount $dir + fi +} +fstab() { + while read -r start mpoint end; do + l="$start $mpoint $end" + # kill off any lines that duplicate the mount point. + sed --follow-symlinks -ri "\%$l%b;\%^\s*\S+\s+$mpoint\s%d" /etc/fstab + tu /etc/fstab <<<"$l" + done +} +pid-check() { + for p in ${pids}; do + for m in ${my_pids[@]}; do + if (( p == m )); then + echo "$0: error: pids to kill includes our pid or a parent. ps output:" >&2 + ps -f -p $p + exit 1 + fi + done + done +} +kill-dir() { + for sig; do + echo kill-dir $sig + found_pids=false + if pids=$(timeout 4 lsof -t $dir); then + found_pids=true + timeout 4 lsof -w $dir + pid-check + kill -$sig $pids + fi + # fuser will find open sockets that lsof won't, for example from gpg-agent. + # note: -v shows kernel processes, which then doesn't return true when we want + if pids=$(timeout 4 fuser -m $dir 2>/dev/null); then + pid-check + found_pids=true + fuser -$sig -mvk $dir fi + sleep .5 + if ! $found_pids; then + return 0 + fi + done + return 1 } +force=false +if [[ $1 == -f ]]; then + force=true +fi + ret=0 ##### begin setup fstab for subvols we care about ###### -first_root_crypt=$(awk '$2 == "/" {print $1}' /etc/mtab) -tu /etc/fstab </dev/null; then + crypt_dev=$root_dev +else # if we are in a recovery boot, find the next best crypt device + noauto=,noauto + for dev in $(dmsetup ls --target crypt | awk '{print $1}'); do + dev=/dev/mapper/$dev + if awk '{print $1}' /etc/mtab | grep -Fx $dev &>/dev/null; then + crypt_dev=$dev + break + fi + done +fi + + +fstab </dev/null; then - continue - fi + d=/$vol + if ! awk '{print $2}' /etc/fstab | grep -xF $d &>/dev/null; then + continue + fi - ##### begin building up list of bind mounts ###### - binds=() # list of bind mounts - roots=($d) # list of bind mounts, plus the original mount - while true; do - new_roots=() - for r in ${roots[@]}; do - # eg. when r=/q/p, for lines like - # /q/p /p none bind 0 0 - # output /p - new_roots+=($(sed -rn "s#^$r/\S+\s+(\S+)\s+none\s+bind\s.*#\1#p" /etc/fstab)) - done - (( ${#new_roots} )) || break - binds+=(${new_roots[@]}) - roots=( ${new_roots[@]} ) + ##### begin building up list of bind mounts ###### + binds=() # list of bind mounts + roots=($d) # list of bind mounts, plus the original mount + while true; do + new_roots=() + for r in ${roots[@]}; do + # eg. when r=/q/p, for lines like + # /q/p /p none bind 0 0 + # output /p + new_roots+=($(sed -rn "s#^$r/\S+\s+(\S+)\s+none\s+bind\s.*#\1#p" /etc/fstab)) done - ##### end building up list of bind mounts ###### + (( ${#new_roots} )) || break + binds+=(${new_roots[@]}) + roots=( ${new_roots[@]} ) + done + ##### end building up list of bind mounts ###### - # if latest is already mounted, make sure binds are mounted and move on - if e check-subvol-stale $d; then - mnt $d - for b in ${binds[@]}; do - mnt $b - done - continue - fi - - fresh_snap=$(/dev/null); then + mnt $d + for b in ${binds[@]}; do + mnt $b + done + continue + fi - umount_ret=true - unmounted=() - for dir in $(echo $d ${binds[*]}\ |tac -s\ ); do - if mountpoint $dir; then - if e umount -R $dir; then - unmounted+=($dir) - else - echo "$0: failed to umount $dir" - # lsof will fail if it finds no pids - if ! e lsof $dir; then - umount_ret=false - ret=1 - continue - fi - pids=$(lsof -t $dir) ||: - kill $pids - sleep .5 - if e umount -R $dir; then - unmounted+=($dir) - else - umount_ret=false - ret=1 - continue - fi - fi + umount_ret=true + unmounted=() + for dir in $(echo $d ${binds[*]}\ |tac -s\ ); do + if mountpoint $dir; then + if e umount -R $dir; then + unmounted+=($dir) + else + if ! kill-dir TERM TERM TERM INT INT HUP HUP; then + if $force; then kill-dir KILL; fi fi - done - if ! $umount_ret; then - for dir in ${unmounted[@]}; do - mnt $dir - done - continue + if e umount -R $dir; then + unmounted+=($dir) + else + echo "$0: failed to umount $dir" + umount_ret=false + ret=1 + continue + fi + fi fi + done - # todo: decipher /mnt/root, like we do in check-subvol-stale - cd /mnt/root - if [[ -e $vol ]]; then - e mv $vol $vol.leaf.$(date +%Y%m%dT%H%M%S%z) - fi - # Note, we make a few assumptions in this script, like - # $d was not a different subvol id than $vol, and - # things otherwise didn't get mounted very strangely. - e btrfs sub snapshot $fresh_snap $vol - for dir in $d ${binds[@]}; do - e mnt $dir + if ! $umount_ret; then + for dir in ${unmounted[@]}; do + mnt $dir + done + continue + fi + + # todo: decipher /mnt/root, like we do in check-subvol-stale + cd /mnt/root + if [[ -e $vol ]]; then + e mv $vol $vol.leaf.$(date +%Y-%m-%dT%H:%M:%S%z) + leaf_vols=($vol.leaf.*) + for leaf in ${leaf_vols[@]}; do + leaf_secs=$(date -d ${leaf#$vol.leaf.} +%s) + if (( $(date +%s) - 60*60*24*60 > leaf_secs )); then # 60 days + e btrfs sub del $leaf + fi done - stale_dir=/nocow/btrfs-stale - rm -f $stale_dir/$d + fi + # Note, we make a few assumptions in this script, like + # $d was not a different subvol id than $vol, and + # things otherwise didn't get mounted very strangely. + e btrfs sub snapshot $fresh_snap $vol + for dir in $d ${binds[@]}; do + e mnt $dir + done + stale_dir=/nocow/btrfs-stale + rm -f $stale_dir/$d done ### disabled -if [[ $HOSTNAME == treetowlxxxxxxxxx ]]; then - # partitioned it with fai partitioner outside of fai, - # because it\'s worth it to have 1% space reserved for boot and - # swap partitions in case I ever want to boot off those drives. - # as root: - # . /a/bin/fai/fai-wrapper - # eval-fai-classfile /a/bin/fai/fai/config/class/51-multi-boot - # fai-setclass ROTATIONAL - # export LUKS_DIR=/q/root/luks/ - # # because the partition nums existed already - # fai-setclass REPARTITION - # /a/bin/fai/fai/config/hooks/partition.DEFAULT - - devs=( - ata-TOSHIBA_MD04ACA500_84REK6NTFS9A-part1 - ata-TOSHIBA_MD04ACA500_84R2K773FS9A-part1 - ata-TOSHIBA_MD04ACA500_8471K430FS9A-part1 - ata-TOSHIBA_MD04ACA500_8481K493FS9A-part1 - ) - first=true - for dev in ${devs[@]}; do - if $first; then - first=false - tu /etc/fstab <