Que dois-je faire pour le code dans Bash, si je veux faire écho *
s à la place des mots de passe (ou même masquer complètement les caractères) lorsque l'utilisateur tape quelque chose en utilisant read
?
Comme Mark Rushakoff l'a souligné, read -s
supprimera l'écho des caractères saisis à l'invite. Vous pouvez utiliser cette fonctionnalité dans le cadre de ce script pour faire écho aux astérisques de chaque caractère tapé:
#!/bin/bash
unset password
Prompt="Enter Password:"
while IFS= read -p "$Prompt" -r -s -n 1 char
do
if [[ $char == $'\0' ]]
then
break
fi
Prompt='*'
password+="$char"
done
echo
echo "Done. Password=$password"
J'ai vraiment aimé la réponse que Wirone a donnée, mais je n'ai pas aimé que le retour en arrière continue de supprimer les caractères même dans l'invite "Entrer le mot de passe:".
J'ai également eu des problèmes où une pression trop rapide sur les touches provoquait l'impression de certains caractères à l'écran ... ce n'est jamais une bonne chose lorsque je demande un mot de passe. =)
Ce qui suit est ma version modifiée de la réponse de Wirone qui résout ces problèmes:
#!/bin/bash
unset PASSWORD
unset CHARCOUNT
echo -n "Enter password: "
stty -echo
CHARCOUNT=0
while IFS= read -p "$Prompt" -r -s -n 1 CHAR
do
# Enter - accept password
if [[ $CHAR == $'\0' ]] ; then
break
fi
# Backspace
if [[ $CHAR == $'\177' ]] ; then
if [ $CHARCOUNT -gt 0 ] ; then
CHARCOUNT=$((CHARCOUNT-1))
Prompt=$'\b \b'
PASSWORD="${PASSWORD%?}"
else
Prompt=''
fi
else
CHARCOUNT=$((CHARCOUNT+1))
Prompt='*'
PASSWORD+="$CHAR"
fi
done
stty echo
echo $PASSWORD
read -s
devrait le mettre en mode silencieux:
-s Silent mode. If input is coming from a terminal, characters are not echoed.
Voir la section read
dans man bash
.
Je voudrais ajouter quelque chose à la solution de Dennis Williamson :
#!/bin/bash
unset password
echo -n "Enter password: "
while IFS= read -p "$Prompt" -r -s -n 1 char
do
# Enter - accept password
if [[ $char == $'\0' ]] ; then
break
fi
# Backspace
if [[ $char == $'\177' ]] ; then
Prompt=$'\b \b'
password="${password%?}"
else
Prompt='*'
password+="$char"
fi
done
Dans l'exemple ci-dessus, le script gère correctement le retour arrière.
Je ne connais pas les étoiles, mais stty -echo est ton ami:
#!/bin/sh
read -p "Username: " uname
stty -echo
read -p "Password: " passw; echo
stty echo
Si vous ne vous souciez pas qu'il soit interactif, vous pouvez simplement le faire
read -s pass
echo "$pass" | sed 's/./*/g'
Cela affichera un * pour chaque caractère du mot de passe saisi après avoir appuyé sur enter.
stty -echo
read something
stty echo
arrêtera l'entrée de l'utilisateur en écho à l'écran pour cette lecture. Selon ce que vous faites avec les invites, vous souhaiterez peut-être ajouter une commande supplémentaire echo
pour générer une nouvelle ligne après la lecture.
Je viens de créer cette fonction spécifique à Bash basée sur les réponses de Dennis Williamson, Wirone et Logan VanCuren:
ask() {
local 'args' 'char' 'charcount' 'Prompt' 'reply' 'silent'
# Basic arguments parsing
while [[ "${1++}" ]]; do
case "${1}" in
( '--silent' | '-s' )
silent='yes'
;;
( '--' )
args+=( "${@:2}" )
break
;;
( * )
args+=( "${1}" )
;;
esac
shift || break
done
if [[ "${silent}" == 'yes' ]]; then
for Prompt in "${args[@]}"; do
charcount='0'
Prompt="${Prompt}: "
reply=''
while IFS='' read -n '1' -p "${Prompt}" -r -s 'char'; do
case "${char}" in
# Handles NULL
( $'\000' )
break
;;
# Handles BACKSPACE and DELETE
( $'\010' | $'\177' )
if (( charcount > 0 )); then
Prompt=$'\b \b'
reply="${reply%?}"
(( charcount-- ))
else
Prompt=''
fi
;;
( * )
Prompt='*'
reply+="${char}"
(( charcount++ ))
;;
esac
done
printf '\n' >&2
printf '%s\n' "${reply}"
done
else
for Prompt in "${args[@]}"; do
IFS='' read -p "${Prompt}: " -r 'reply'
printf '%s\n' "${reply}"
done
fi
}
Il pourrait être utilisé comme:
$ ask Username
Username: AzureDiamond
AzureDiamond
$ ask -s Password
Password: *******
hunter2
$ ask First Second Third
First: foo
foo
Second: bar
bar
Third: baz
baz