概述

很多软件不提供init脚本,或者提供的脚本不合胃口,难免要自己动手丰衣足食下面就推荐一个用来启动守护进程的神器
start-stop-daemon是OpenRC计划的一部分,这个程序最先出现在Debian系的Linux发行版中,这里有个比较古老的手册页面,更详细更直观的办法当然是通过man start-stop-daemon来查看手册了我使用的是"start-stop-daemon (OpenRC) gentoo linux"版本,大部分功能是差不多的

常用参数

start-stop-daemon最基本的两个功能就是–start和–stop,简写为-S和-K,然后再加上一个-s|–signal来给进程发送信号,功德圆满
至于其中比较常用的一些参数,我列出来参考一下,以免忘了:

参数 描述
* -x, --exec daemon daemon就是真正要执行的进程脚本,比方说启动nginx,那么就是start-stop-daemon -x nginx
* -p, --pidfile pidfile 指定pid文件,至于pid文件的用途就多了,stop,status都少不了它
* -n, --name 如果没有指定pid文件,那么就要通过指定name来停止进程了
* -u, --user user[:group] 指定脚本用哪个用户或用户组执行,init脚本是必须使用root权限来执行的,但是它fork出来的子进程我们一般会选择一个权限较低的用户
* -b, --background 强制脚本在后台执行
* -m, --make-pidfile 这个一般和-b配合,用于生成pid文件
* -d, --chdir path 切换进程的主目录,这个在构建守护进程的时候是很常用的
* -r, --chroot path 在某些安全性要求较高的情况下,我们就需要用到chroot将进程工作环境与物理环境完全隔离开来
* -1, --stdout logfile 将标准输出记录到log文件
* -2, --stderr logfile 将标准错误输出记录到log文件
* -w, --wait milliseconds 进程启动后,有这个参数会等待几毫秒来检测进程是否仍然存活

参数说完,下面就是一些需要注意的地方了

-b与守护进程
-b是一个很常用的参数,我们使用start-stop-daemon的目的就是为了实现守护进程但是有些程序自身也实现了守护进程的功能,比方说mongodb中有一个fork选项就是将自己在后台执行,这个时候假如搭配的-b参数,是得不到正确的pid的,因为start-stop-daemon只能得到最初启动的父进程pid,而父进程在fork完之后就自动退出了,那么start-stop-daemon就永远找不到正确的pid来结束进程了所以使用-b的时候,一定要保证程序是在前台运行的

其他参数
-x daemon后面跟的执行脚本必须只能是一个文件名,有些程序运行时还需要指定一些参数,比如nginx -c file来指定nginx的配置文件,使用start-stop-daemon -x "nginx -c file"是会报错的,这些程序内的参数以另一种方式加载,start-stop-daemon -x daemon – $ARGV,这里的双横线–后面跟的所有参数就会被带到程序中了,比如start-stop-daemon -x nginx -c /etc/nginx.conf

例子

nodejs的项目保存在/root/gitm/目录下,实际上cd /root/gitm/ && node main.js就可以启动,但是为了实现开机自运行,可以用下边的脚本处理
下面是nodejs自定义web服务的一个init脚本,用start-stop-daemon是非常简单的

#!/sbin/openrc-run

pidfile=/var/run/gitlist.pid
logfile=/var/log/gitlist
name="git project list web server daemon"

mainfile=/root/gitm/main.js

depend(){
    need dhcpcd
}
start(){

    ebegin "start git list daemon"
    start-stop-daemon --start   \
    --make-pidfile              \
    --background                \
    --pidfile ${pidfile}        \
    --stdout ${logfile}         \
    --stderr ${logfile}         \
    --chdir /root/gitm/         \
    --exec node                 \
    -- ${mainfile}
    eend $?
}

stop(){
    ebegin 'stop git list daemon'
        start-stop-daemon --stop --pidfile ${pidfile}
    eend 0
}