SIGINT取消bash脚本中的读取?

时间:2012-10-24 21:08:47

标签: bash sigint

我正在编写一个bash包装器来学习一些脚本概念。我们的想法是在bash中编写一个脚本,并在登录时将其设置为用户的shell。

我做了一个readeval用户输入的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

2 个答案:

答案 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或手动退出。