如何等待脚本生成的所有子(和孙等)进程

Ram*_*Ram 8 unix bash shell process

语境:

用户为我提供了自定义脚本来运行.这些脚本可以是任何类型的脚本,以启动多个GUI程序,后端服务.我无法控制脚本的编写方式.这些脚本可以是阻塞类型,即执行等待直到所有子进程(按顺序运行的程序)退出

#exaple of blocking script
echo "START"
first_program 
second_program 
echo "DONE"
Run Code Online (Sandbox Code Playgroud)

或非阻塞类型,即在后台派生子进程并退出类似的东西

#example of non-blocking script
echo "START"
first_program &
second_program &
echo "DONE"
Run Code Online (Sandbox Code Playgroud)

我想要实现的目标是什么?

用户提供的脚本可以是上述两种类型中的任何一种,也可以是两者的混合.我的工作是运行脚本并等待它启动的所有进程退出然后关闭节点.如果它的阻塞类型,大小写很简单,即得到脚本执行过程的PID并等到ps -ef | grep -ef PID没有更多的条目.非阻塞脚本是给我带来麻烦的

Is there a way I can get list of PIDs of all the child process spawned by execution of a script? Any pointers or hints will be highly appreciated

Ans*_*ers 7

ps --ppid $PID将列出该过程的所有子进程$PID.


che*_*ner 7

您可以使用wait等待所有后台进程userscript完成.由于wait仅适用于当前shell的子项,因此您需要获取其脚本而不是将其作为单独的进程运行.

( source userscript; wait )
Run Code Online (Sandbox Code Playgroud)

在显式子shell中获取脚本应该模拟足够紧密地开始新进程.如果没有,您还可以设置子shell,这会强制启动新进程,然后等待完成.

( source userscript; wait ) & wait
Run Code Online (Sandbox Code Playgroud)


tha*_*guy 5

您可以打开一个被其他进程继承的文件描述符,然后等到它不再被使用。这是一种低开销的方法,通常可以正常工作,但如果进程需要,可以解决它:

 foo=$(mktemp)
 ( flock -x 5000; theirscript; ) 5000> "$foo"
 flock -x 0 < "$foo"
 rm "$foo"
 echo "The script and its subprocesses are done"
Run Code Online (Sandbox Code Playgroud)

您可以使用 ptrace 跟踪所有调用的进程,例如使用strace. 这更容易,但有一些相关的开销,并且在脚本调用 suid 二进制文件时可能不起作用:

strace -f -e none theirscript
Run Code Online (Sandbox Code Playgroud)