How can I prompt for yes/no style confirmation in

2020-06-03 05:22发布

问题:

Using zsh, I'm trying to put a step in my ~/.zprofile where I interactively ask a yes/no style question. At first I tried this bash-style approach, but I saw errors of this form:

read: -p: no coprocess

(I'm aware that typically the zsh syntax is different from bash's - I tried preceding it with a sh emulation command - emulate -LR sh - but it made no difference).

This page implied the syntax might be different, so guided by this page and the zsh man page, I tried this instead:

read -q REPLY?"This is the question I want to ask?"

This instead fails with an error of the form:

/home/user/.zprofile:5: no matches found: REPLY?"This is the question I want to ask?"

How can I ask a simple yes/no question with zsh? Ideally the command would just swallow one character, with no need to press Enter/Return, and be 'safe' - i.e. the subsequent test defaults to no/false unless 'Y' or 'y' are entered.

回答1:

From zsh - read

If the first argument contains a ‘?’, the remainder of this word is used as a prompt on standard error when the shell is interactive.

You must quote the entire argument

read -q "REPLY?This is the question I want to ask?"

this will prompt you with This is the question I want to ask? and return the character pressed in REPLY.

If you don't quote the question mark, zsh tries to match the argument as a filename. And if it doesn't find any matching filename, it complains with no matches found.



回答2:

See ZSH Manual for documentation of ZSH's read. Try:

read REPLY\?"This is the question I want to ask?"


回答3:

I'm adding this answer because every time you want to ask the user for confirmation, you also want to act on it. here's a function that prompts with read -q (thanks, other answers!) and branches on the result to do what you want (in this case, git stuff):

git_commit_and_pull() {
    # http://zsh.sourceforge.net/Doc/Release/Shell-Builtin-Commands.html#index-read
    if read -q "choice?Press Y/y to continue with commit and pull: "; then
        set -x
        git add . && git commit -m 'haha git goes brrr' && git pull
        { set +x; } 2>/dev/null
    else
        echo
        echo "'$choice' not 'Y' or 'y'. Exiting..."
    fi
}