POSIX替代bash读取超时和字符限制

时间:2015-08-25 20:47:59

标签: shell posix

我正在编写一个交互式shell脚本,需要在尽可能多的系统上运行。是否有另一种方法可以实现与标准POSIX系统兼容的以下方法?

#! /bin/bash
echo -n "Do you wish to continue? (Y/n) 5 seconds to respond... "
read -n 1 -t 5 answer # accepts a single character, 5 second timeout.
if [ "$answer" = "n" ] || [ "$answer" = "N" ] ; then
  echo -e "\nExiting..."
  exit
fi
echo -e "\nContinuing with script..."
# More code

read上的超时对我来说最重要(read -t 5)。读取的一个字符限制是可取的,但不是必需的(read -n 1)。 理想情况下,该脚本可以在POSIX系统上运行,也可以在bash中运行,而无需启用特殊的POSIX兼容模式。

2 个答案:

答案 0 :(得分:4)

根据Dickey先生的回答调整stty设置,如果按下“n”或“N”以外的任何内容,以下内容似乎可以正常工作并继续使用。

据我所知,所有stty设置都是posix。

#!/bin/sh

read_char() {
        old=$(stty -g)
        stty raw -echo min 0 time 50
        eval "$1=\$(dd bs=1 count=1 2>/dev/null)"
        stty $old
}

printf "Do you wish to continue? (Y/n) 5 seconds to respond..."
read_char answer
# answer=$(getch)
# answer=$(getche)

if [ "$answer" = "n" ] || [ "$answer" = "N" ] ; then
  printf "\nExiting...\n"
  exit
fi

printf "\nContinuing with script...\n"

“read_char”的替代选项:

这避免了使用eval(可能不安全)

getch() {
        old=$(stty -g)
        stty raw -echo min 0 time 50
        printf '%s' $(dd bs=1 count=1 2>/dev/null)
        stty $old
}

这可以避免使用eval并打印按下的键

getche() {
        old=$(stty -g)
        stty raw min 0 time 50
        printf '%s' $(dd bs=1 count=1 2>/dev/null)
        stty $old
}

答案 1 :(得分:1)

stty程序提供了执行此操作的方法。 xterm有几个脚本(在其源代码的“vttests”子目录中),它保存,修改和恢复终端设置,以允许它读取终端对转义序列的响应。以下是dynamic2.sh(开头设置printfecho的一部分,用于解决一些带有$CMD变量的旧系统:

echo "reading current color settings"

exec </dev/tty
old=`stty -g`
stty raw -echo min 0  time 5

original=
for N in $FULL
do
    $CMD $OPT "${ESC}]$N;?^G${SUF}" > /dev/tty
    read reply
    eval original$N='${reply}${SUF}'
    original=${original}${reply}${SUF}
done
stty $old

if ( trap "echo exit" EXIT 2>/dev/null ) >/dev/null
then
    trap '$CMD $OPT "$original" >/dev/tty; exit' EXIT HUP INT TRAP TERM
else
    trap '$CMD $OPT "$original" >/dev/tty; exit' 0    1   2   5    15
fi