web-dev-qa-db-fra.com

Comment puis-je faire écho aux étoiles (*) lors de la lecture du mot de passe avec `read`?

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?

37
Deniz Zoeteman

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"
58
Dennis Williamson

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
19
Logan VanCuren

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 .

12
Mark Rushakoff

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.

Source

6
Wirone

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

Source: http://www.peterbe.com/plog/passwords-with-bash

5
Vinko Vrsalovic

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.

2
Dino Dini
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.

2
moonshadow

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
1
nxnev