【发布时间】:2011-09-06 00:45:49
【问题描述】:
从提示符中获取后台进程 ID 很容易,方法是:
$ my_daemon &
$ echo $!
但是如果我想以不同的用户身份运行它怎么办:
su - joe -c "/path/to/my_daemon &;"
现在如何获取 my_daemon 的 PID?
【问题讨论】:
-
在我看来,这对我来说确实是主题;这是一个关于 shell 编程的问题,因此在 SO 的范围内。
从提示符中获取后台进程 ID 很容易,方法是:
$ my_daemon &
$ echo $!
但是如果我想以不同的用户身份运行它怎么办:
su - joe -c "/path/to/my_daemon &;"
现在如何获取 my_daemon 的 PID?
【问题讨论】:
简明扼要 - 有很多困难。
您必须安排 su'd shell 将子 PID 写入文件,然后选择输出。鉴于创建文件的是“joe”而不是“dex”,这就增加了另一层复杂性。
最简单的解决办法大概是:
su - joe -c "/path/to/my_daemon & echo \$! > /tmp/su.joe.$$"
bg=$(</tmp/su.joe.$$)
rm -f /tmp/su.joe.$$ # Probably fails - joe owns it, dex does not
下一个解决方案涉及使用备用文件描述符 - 编号 3。
su - joe -c "/path/to/my_daemon 3>&- & echo \$! 1>&3" 3>/tmp/su.joe.$$
bg=$(</tmp/su.joe.$$)
rm -f /tmp/su.joe.$$
如果您担心中断等(您可能应该担心),那么您也可以捕获一些东西:
tmp=/tmp/su.joe.$$
trap "rm -f $tmp; exit 1" 0 1 2 3 13 15
su - joe -c "/path/to/my_daemon 3>&- & echo \$! 1>&3" 3>$tmp
bg=$(<$tmp)
rm -f $tmp
trap 0 1 2 3 13 15
(捕获的信号是 HUP、INT、QUIT、PIPE 和 TERM - 加上 0 表示 shell 退出。)
警告:不错的理论 - 未经测试的代码...
【讨论】:
su - joe -c "/path/to/my_daemon & echo \$! > /tmp/su.joe.$$" 逃避 $! my_daemon & 之后没有分号当然也是陷阱。我打算玩一下。
这里介绍的方法对我不起作用。这是我所做的:
PID_FILE=/tmp/service_pid_file
su -m $SERVICE_USER -s /bin/bash -c "/path/to/executable $ARGS >/dev/null 2>&1 & echo \$! >$PID_FILE"
PID=`cat $PID_FILE`
【讨论】:
只要后台进程的输出被重定向,就可以将PID发送到stdout:
su "${user}" -c "${executable} > '${log_file}' 2>&1 & echo \$!"
然后可以将 PID 重定向到第一个用户拥有的文件,而不是第二个用户。
su "${user}" -c "${executable} > '${log_file}' 2>&1 & echo \$!" > "${pid_file}"
不过,日志文件确实需要由第二个用户拥有才能这样做。
【讨论】:
这是我的解决方案
su oracle -c "/home/oracle/database/runInstaller" &
pid=$(pgrep -P $!)
解释
pgrep -P $! - 获取父 pid $! 的子进程
【讨论】:
我在Linux上采用了上述解决方案,但不得不添加一个睡眠来让子进程有机会启动。
su - joe -c "/path/to/my_daemon > /some/output/file" &
parent=$!
sleep 1
pid=$(pgrep -P $parent)
在 bash 中运行,它不喜欢 pid=$(pgrep -P $!),但如果我在 ! 之后添加一个空格,则可以:pid=$(pgrep -P $! )。我坚持使用额外的 $parent 变量来提醒自己下次查看脚本时我在做什么。
【讨论】: