crypt: functions for ask-for-password and reading key

Asking for password and reading key parts are moved to separate
functions in crypt-lib.sh: ask_for_password and readkey.
master
Amadeusz Żołnowski 2011-05-05 13:34:20 +02:00 committed by Harald Hoyer
parent 2f95d699dc
commit 3909d7edfc
2 changed files with 113 additions and 31 deletions

View File

@ -4,6 +4,78 @@


command -v getarg >/dev/null || . /lib/dracut-lib.sh command -v getarg >/dev/null || . /lib/dracut-lib.sh


# ask_for_password
#
# Wraps around plymouth ask-for-password and adds fallback to tty password ask
# if plymouth is not present.
#
# --cmd command
# Command to execute. Required.
# --prompt prompt
# Password prompt. Note that function already adds ':' at the end.
# Recommended.
# --tries n
# How many times repeat command on its failure. Default is 3.
# --ply-[cmd|prompt|tries]
# Command/prompt/tries specific for plymouth password ask only.
# --tty-[cmd|prompt|tries]
# Command/prompt/tries specific for tty password ask only.
# --tty-echo-off
# Turn off input echo before tty command is executed and turn on after.
# It's useful when password is read from stdin.
ask_for_password() {
local cmd; local prompt; local tries=3
local ply_cmd; local ply_prompt; local ply_tries=3
local tty_cmd; local tty_prompt; local tty_tries=3
local ret

while [ $# -gt 0 ]; do
case "$1" in
--cmd) ply_cmd="$2"; tty_cmd="$2" shift;;
--ply-cmd) ply_cmd="$2"; shift;;
--tty-cmd) tty_cmd="$2"; shift;;
--prompt) ply_prompt="$2"; tty_prompt="$2" shift;;
--ply-prompt) ply_prompt="$2"; shift;;
--tty-prompt) tty_prompt="$2"; shift;;
--tries) ply_tries="$2"; tty_tries="$2"; shift;;
--ply-tries) ply_tries="$2"; shift;;
--tty-tries) tty_tries="$2"; shift;;
--tty-echo-off) tty_echo_off=yes;;
esac
shift
done

{ flock -s 9;
# Prompt for password with plymouth, if installed and running.
if [ -x /bin/plymouth ] && /bin/plymouth --has-active-vt; then
/bin/plymouth ask-for-password \
--prompt "$ply_prompt" --number-of-tries=$ply_tries \
--command="$ply_cmd"
ret=$?
else
if [ "$tty_echo_off" = yes ]; then
stty_orig="$(stty -g)"
stty -echo
fi

local i=1
while [ $i -le $tty_tries ]; do
[ -n "$tty_prompt" ] && \
printf "$tty_prompt [$i/$tty_tries]:" >&2
eval "$tty_cmd" && ret=0 && break
ret=$?
i=$(($i+1))
[ -n "$tty_prompt" ] && printf '\n' >&2
done

[ "$tty_echo_off" = yes ] && stty $stty_orig
fi
} 9>/.console.lock

[ $ret -ne 0 ] && echo "Wrong password" >&2
return $ret
}

# Try to mount specified device (by path, by UUID or by label) and check # Try to mount specified device (by path, by UUID or by label) and check
# the path with 'test'. # the path with 'test'.
# #
@ -118,3 +190,33 @@ getkey() {


return 1 return 1
} }

# readkey keypath keydev device
#
# Mounts <keydev>, reads key from file <keypath>, optionally processes it (e.g.
# if encrypted with GPG) and prints to standard output which is supposed to be
# read by cryptsetup. <device> is just passed to helper function for
# informational purpose.
readkey() {
local keypath="$1"
local keydev="$2"
local device="$3"

local mntp=$(mkuniqdir /mnt keydev)
mount -r "$keydev" "$mntp" || die 'Mounting rem. dev. failed!'

case "${keypath##*.}" in
gpg)
if [ -f /lib/dracut-crypt-gpg-lib.sh ]; then
. /lib/dracut-crypt-gpg-lib.sh
gpg_decrypt "$mntp" "$keypath" "$keydev" "$device"
else
die "No GPG support to decrypt '$keypath' on '$keydev'."
fi
;;
*) cat "$mntp/$keypath" ;;
esac

umount "$mntp"
rmdir "$mntp"
}

View File

@ -76,38 +76,18 @@ if [ -n "$(getarg rd.luks.key)" ]; then
fi fi
unset tmp unset tmp


mntp=$(mkuniqdir /mnt keydev) info "Using '$keypath' on '$keydev'"
mount -r "$keydev" "$mntp" || die 'Mounting rem. dev. failed!' readkey "$keypath" "$keydev" "$device" \
cryptsetup -d "$mntp/$keypath" luksOpen "$device" "$luksname" | cryptsetup -d - luksOpen "$device" "$luksname"
umount "$mntp" unset keypath keydev
rmdir "$mntp"
unset mntp keypath keydev
else else
# Prompt for password with plymouth, if installed and running. luks_open="$(command -v cryptsetup) luksOpen"
if [ -x /bin/plymouth ] && /bin/plymouth --has-active-vt; then ask_for_password --ply-tries 5 \
prompt="Password [$device ($luksname)]:" --ply-cmd "$luks_open -T1 $device $luksname" \
if [ ${#luksname} -gt 8 ]; then --ply-prompt "Password ($device)" \
sluksname=${sluksname##luks-} --tty-tries 1 \
sluksname=${luksname%%${luksname##????????}} --tty-cmd "$luks_open -T5 $device $luksname"
prompt="Password for $device ($sluksname...)" unset luks_open
fi
# flock against other interactive activities
{ flock -s 9;
/bin/plymouth ask-for-password \
--prompt "$prompt" --number-of-tries=5 \
--command="$(command -v cryptsetup) luksOpen -T1 $device $luksname"
} 9>/.console.lock
unset sluksname prompt
else
# flock against other interactive activities
{ flock -s 9;
echo "$device ($luksname) is password protected"
cryptsetup luksOpen -T5 $device $luksname
} 9>/.console.lock
fi
fi fi


unset device luksname unset device luksname