当后台 zenity 对话框关闭时使脚本终止
Make script terminate when background zenity dialog is closed
我在这里编写了一个相当简单的脚本,用于使用 zenity
显示文本信息对话框,并不断从远程 TCP 连接读取数据并将其显示在对话框中。这有效...但是,如果我关闭 zenity
对话框,我希望整个脚本终止。
有办法吗?我不认为我可以检查 while 循环中的任何内容,因为脚本可能会在从远程 TCP 连接读取数据时停止。
#!/bin/bash
on_exit() {
zenity --display=:0 --error --text="Script has exited." &
}
# Options
while getopts "a:p:t:" OPTION; do case "$OPTION" in
a) address="$OPTARG";;
p) port="$OPTARG";;
t) title="$OPTARG";;
esac; done
exec &> >(zenity --display=:0 --text-info --title=$title || exit)
# doesn't make a difference? ↑
# also tried &&
trap "on_exit" EXIT
while read data < /dev/tcp/$address/$port; do
echo $data
# ...
# do some other stuff with the information
# ...
done
注意:这将是 运行 在 IGEL Linux 上。我没有安装额外软件包的选项。因此,理想情况下,我正在寻找的解决方案是 Bash 的原生解决方案。
更新
我只需要进行此修改即可继续使用 exec
。或者@BachLien 使用命名管道的回答也有效。
PID=$$
exec &> >(zenity --display=:0 --text-info --title=$title; kill $PID)
我没有安装 zenity
,所以我尝试了这个脚本来说明这个想法。
程序terminal+cat
(模拟zenity
)由后台运行的函数_dspMsg
执行(子进程); cat
连续显示来自文件 ($m
) 的消息,该文件是一个命名管道;当 terminal+cat
退出时,父进程被 kill
ed。
与此同时,另一个 cat
进程将消息写入管道 $m
(模拟 TPC 信息源);当 _dspMsg
退出时它会被杀死。
#!/bin/bash
# 1) named pipe
m=`mktemp -u /tmp/msg-XXXX=` # get a temporary filename (for named pipe)
mkfifo "$m" # create that named pipe
trap "echo END; rm $m" EXIT # remove that file when exit
# 2) zenity
_dspMsg(){ # continuously display messages
urxvt -e bash -c "cat <$m" # terminal+cat is used in place of zenity
kill # kill parent pid
} # to be run in background
_dspMsg $$ & # $$ = proccess id
# 3) TCP info feeds
cat >>"$m" # feeding messages using cat
# cat is used in placed of TCP data feed
注:
- 命名管道用作父进程和子进程之间的通信方式。
- 要测试该脚本,您可能需要将
urxvt
更改为 xterm
、iTerm
或您计算机上可用的任何其他终端仿真器。
所以,也许这就是您所需要的(未经测试):
#!/bin/bash
while getopts "a:p:t:" OPTION; do case "$OPTION" in
a) address="$OPTARG";;
p) port="$OPTARG";;
t) title="$OPTARG";;
esac; done
m=`mktemp -u /tmp/msg-XXXX=`
mkfifo "$m"
trap "zenity --display=:0 --error --text='Script has exited.' & rm $m" EXIT
_dspMsg(){
zenity --display=:0 --text-info --title="$title" <"$m"
kill
}
_dspMsg $$ &
while read data < /dev/tcp/$address/$port; do
echo $data >>"$m"
done
您可以将 while
循环的所有输出通过管道传输到 zenity,从而摆脱对 exec &>
.
的需要
while read data < "/dev/tcp/$address/$port"; do
echo "$data"
done | zenity --display=:0 --text-info --title="$title"
我在这里编写了一个相当简单的脚本,用于使用 zenity
显示文本信息对话框,并不断从远程 TCP 连接读取数据并将其显示在对话框中。这有效...但是,如果我关闭 zenity
对话框,我希望整个脚本终止。
有办法吗?我不认为我可以检查 while 循环中的任何内容,因为脚本可能会在从远程 TCP 连接读取数据时停止。
#!/bin/bash
on_exit() {
zenity --display=:0 --error --text="Script has exited." &
}
# Options
while getopts "a:p:t:" OPTION; do case "$OPTION" in
a) address="$OPTARG";;
p) port="$OPTARG";;
t) title="$OPTARG";;
esac; done
exec &> >(zenity --display=:0 --text-info --title=$title || exit)
# doesn't make a difference? ↑
# also tried &&
trap "on_exit" EXIT
while read data < /dev/tcp/$address/$port; do
echo $data
# ...
# do some other stuff with the information
# ...
done
注意:这将是 运行 在 IGEL Linux 上。我没有安装额外软件包的选项。因此,理想情况下,我正在寻找的解决方案是 Bash 的原生解决方案。
更新
我只需要进行此修改即可继续使用 exec
。或者@BachLien 使用命名管道的回答也有效。
PID=$$
exec &> >(zenity --display=:0 --text-info --title=$title; kill $PID)
我没有安装 zenity
,所以我尝试了这个脚本来说明这个想法。
程序terminal+cat
(模拟zenity
)由后台运行的函数_dspMsg
执行(子进程); cat
连续显示来自文件 ($m
) 的消息,该文件是一个命名管道;当 terminal+cat
退出时,父进程被 kill
ed。
与此同时,另一个 cat
进程将消息写入管道 $m
(模拟 TPC 信息源);当 _dspMsg
退出时它会被杀死。
#!/bin/bash
# 1) named pipe
m=`mktemp -u /tmp/msg-XXXX=` # get a temporary filename (for named pipe)
mkfifo "$m" # create that named pipe
trap "echo END; rm $m" EXIT # remove that file when exit
# 2) zenity
_dspMsg(){ # continuously display messages
urxvt -e bash -c "cat <$m" # terminal+cat is used in place of zenity
kill # kill parent pid
} # to be run in background
_dspMsg $$ & # $$ = proccess id
# 3) TCP info feeds
cat >>"$m" # feeding messages using cat
# cat is used in placed of TCP data feed
注:
- 命名管道用作父进程和子进程之间的通信方式。
- 要测试该脚本,您可能需要将
urxvt
更改为xterm
、iTerm
或您计算机上可用的任何其他终端仿真器。
所以,也许这就是您所需要的(未经测试):
#!/bin/bash
while getopts "a:p:t:" OPTION; do case "$OPTION" in
a) address="$OPTARG";;
p) port="$OPTARG";;
t) title="$OPTARG";;
esac; done
m=`mktemp -u /tmp/msg-XXXX=`
mkfifo "$m"
trap "zenity --display=:0 --error --text='Script has exited.' & rm $m" EXIT
_dspMsg(){
zenity --display=:0 --text-info --title="$title" <"$m"
kill
}
_dspMsg $$ &
while read data < /dev/tcp/$address/$port; do
echo $data >>"$m"
done
您可以将 while
循环的所有输出通过管道传输到 zenity,从而摆脱对 exec &>
.
while read data < "/dev/tcp/$address/$port"; do
echo "$data"
done | zenity --display=:0 --text-info --title="$title"