我正在编写一个bash包装器来学习一些脚本概念。我们的想法是在bash中编写一个脚本,并在登录时将其设置为用户的shell。
我做了一个read
和eval
用户输入的while循环,然后注意到,每当用户键入CTRL + C
时,脚本都会中止,因此用户会话结束。
为了避免这种情况,我陷入SIGINT
,在陷阱中什么都不做。
现在,问题在于,当您在命令的一半处键入CTRL + C
时,它不会像在bash上那样被取消 - 它只会忽略CTRL + C
。
因此,如果我输入ping stockoverf^Cping stackoverflow.com
,我会获得ping stockoverfping stackoverflow.com
而不是我想要的ping stackoverflow.com
。
有没有办法做到这一点?
#!/bin/bash
# let's trap SIGINT (CTRL + C)
trap "" SIGINT
while true
do
read -e -p "$USER - SHIELD: `pwd`> " command
history -s $command
eval $command
done
答案 0 :(得分:3)
您可以使用像xdotool这样的工具发送Ctrl-A(行首)Ctrl-K(删除到行尾)返回(清理行)
#!/bin/bash
trap "xdotool key Ctrl+A Ctrl+k Return" SIGINT;
unset command
while [ "$command" != "quit" ] ;do
eval $command
read -e -p "$USER - SHIELD: `pwd`> " command
done
trap SIGINT
但我强烈邀请你去rtfm ...在搜索``debug''关键字......
man -Pless\ +/debug bash
答案 1 :(得分:1)
我知道这很老了,但我一直在努力做这样的事情,并提出了解决方案。希望它可以帮助其他人!
#/usr/bin/env bash
# Works ok when it is invoked as a bash script, but not when sourced!
function reset_cursor(){
echo
}
trap reset_cursor INT
while true; do
command=$( if read -e -p "> " line ; then echo "$line"; else echo "quit"; fi )
if [[ "$command" == "quit" ]] ; then
exit
else
history -s $command
eval "$command"
fi
done
trap SIGINT
通过将读取结果放入子外壳中,可以确保读取结果被sigint信号杀死。如果您在该sigint渗透到父级时捕获了该sigint,则可以在那里忽略它,并进入下一个while循环。您不必具有reset_cursor作为其自身的功能,但是如果您想做更复杂的事情,我觉得很好。
我必须在子外壳中添加if语句,因为否则它将忽略ctrl + d-但我们希望它能够“注销我们”而不会强迫用户键入exit或手动退出。