Как отправить сигнал SIGINT из скрипта в скрипт?
Я хочу перехватить сигнал, отправленный из Script-A.sh в Script-B.sh
поэтому в Script-A.sh я использую команду:
(Send SIGINT to Script-B.sh)
kill -2 $PID_Script-B.sh
А в Script-B.sh я ловлю сигнал и вызываю функцию Clean
ловушка "Чистый" 2
Это не работает, вместо этого Script-B.sh убивается сразу, не выполняя Clean !!
Также я заметил, что если я хочу отправить SIGINT из терминала на любой скрипт, который его перехватывает, ctrl-c
будет перехвачен правильно, но если я укажу сигнал с помощью команды kill -2 $pid_of_script
Любая идея о разнице между двумя способами отправки SIGINT (ctrl-c
VS kill -2 $pid_of_script
), и как я могу отправить SIGINT из сценария в другой?
Ответы
Ответ 1
Я смог воспроизвести поведение, о котором вы сообщаете. Моя гипотеза заключается в том, что, поскольку script работает от, неинтерактивная оболочка (в качестве дочернего элемента script), которая SIGINT
, которая является сигналом клавиатуры, игнорируется.
От info bash
:
Фоновые процессы - это те, чей идентификатор группы процессов отличается от терминал; такие процессы невосприимчивы к сигналам, генерируемым клавиатурой.
Я обнаружил, что если вы trap
и kill
используете другой сигнал, например SIGUSR1
, он работает.
Дополнительная информация от man bash
:
Нестроенные команды, выполняемые bash, имеют обработчики сигналов, установленные для значений, унаследованных оболочкой от родителя. Когда управление заданиями не действует, асинхронные команды игнорируют SIGINT и SIGQUIT в дополнение к этим унаследованным обработчикам.
и
Если bash ожидает завершения команды и получает сигнал, для которого установлена ловушка, ловушка не будет выполняться до тех пор, пока команда не завершится.
и
Любая ловушка в SIGCHLD выполняется для каждого дочернего элемента, который завершает работу.
Ответ 2
В script A: Функция Trap будет выглядеть следующим образом, которая вызовет функцию trap_mesg() в scriptA.sh. Сигнал KILL (2/INTerrupt, 5/TERMinate-default). Все, что вам нужно сделать, это получить PID исполняемого скриптаB.sh процесса/сеанса, как только scriptB.sh вызывается из scriptA.sh(nohup... и даст вам или использует команду ps)
trap_mesg ()
{
#...do something here for script B..
# i.e.
kill -2 PID_of_ScriptB.sh_running_process_session
sleep 10; #just in case, not reqd though.
#show using ps -eAf|grep "scriptB" ... if null means, scriptB is gone. user is happy now.
#...before actually exiting out...
#show script A is exiting out as ScriptB is dead already, time for scriptA now.
#...do something here..
}
#####################################
## Trap signals : INT, TERM. catch ##
#####################################
#Set NULL/Blank value to trap_call. This variable will help in running trap_mesg only once during the life of this script.
trap_call="";
trap 'if [ -z ${trap_call} ]; then trap_call="1"; trap_mesg ; fi' 2 15
##################################
Теперь, в scriptB.sh, выполните одно и то же/подобное, но только для задания trapBash (например, вызов clean).
clean ()
{
echo "karoge seva to milega meva";
rm -fr /some/folder_file
}
trap_mesg ()
{
#...do something here JUST for script B trap message work..
# i.e.
clean;
#...do something here..
}
#####################################
## Trap signals : INT, TERM. catch ##
#####################################
#Set NULL/Blank value to trap_call. This variable will help in running trap_mesg only once during the life of this script.
trap_call="";
trap 'if [ -z ${trap_call} ]; then trap_call="1"; trap_mesg ; fi' 2 15
##################################
Таким образом, вам не нужно ссылаться на scriptB.sh в scriptA.sh как ". scriptB.sh...."