You are not logged in.
I'm trying to have a yes/no prompt which accepts y, n, yes, no, and is case insensitive. I thought this would work:
until [[ "$yesno" =~ no?|y(es)? ]]
do
read "Continue (yes/no)?" reply;
yesno=`tr '[:upper:]' '[:lower:]' <<< "$reply"`;
if [[ "$yesno" =~ no? ]]
then
exit 0;
elif [[ "$yesno" =~ y(es)? ]]
then
# stuff to do
else
"Invalid input!";
done
The problem is ? seems to act more like a wildcard. I tested yn, and it matches. In fact as long as the first character is a Y, y, N, or n, it returns true. The only time ? acts the way it should (match preceding character once or none), is when another character follows it:
# 'non' and 'noun' will match
if [[ "$1" =~ nou?n ]]
then
echo "$1 matches";
else
echo "$1 does not match";
fi
I would like to know why it's working like this, and how do I fix it?
Offline
i think you should use ^no?$ and ^y(es)?$
This silver ladybug at line 28...
Offline
Yup, the issue is that you're only specifying y(es)? should appear somewhere in the string, not that it is the whole string - lolilolicon's suggestion should fix it.
Offline
You could use the "case" conditional construct: http://www.gnu.org/software/bash/manual … Constructs
Love, and do what thou wilt - St. Augustine of Hippo
Offline
Instead of calling an external program, tr, you could let bash do the case conversion with parameter expansion:
yesno=${reply,,}
See 3.5.3 Shell Parameter Expansion. This is more compact but more cryptic. Take your choice.
Or you could obviate the need for case conversion altogether by having bash do case-insensitive comparisons:
shopt -s nocasematch
But then, to prevent the nocasematch setting affecting the rest of the script, you'd have to save the old value of nocasematch and reset it when you've finished, or carry out the comparisons in a subshell (in which case any other variable settings done in the subshell will also be lost when the subshell ends), so I think it would be more hassle than it's worth.
Last edited by nh (2009-11-16 17:16:04)
Offline