Daemon init script for node.js based app/server (DEBIAN/UBUNTU): различия между версиями
Материал из support.qbpro.ru
imported>Vix Нет описания правки |
imported>Supportadmin |
||
(не показано 10 промежуточных версий 2 участников) | |||
Строка 59: | Строка 59: | ||
DAEMON_ARGS="/path/to/app.js" # path to your node.js server/app | DAEMON_ARGS="/path/to/app.js" # path to your node.js server/app | ||
# NB: don't use ~/ in path | |||
DESC="node.js http server" # whatever fancy description you like | DESC="node.js http server" # whatever fancy description you like | ||
NODEUSER=myuser:mygroup # USER who OWNS the daemon process (no matter whoever runs the init script) | NODEUSER=myuser:mygroup # USER who OWNS the daemon process (no matter whoever runs the init script) | ||
# user:group (if no group is specified, the primary GID for that user is used) | |||
LOCAL_VAR_RUN=/usr/local/var/run # in case the init script is run by non-root user, you need to | LOCAL_VAR_RUN=/usr/local/var/run # in case the init script is run by non-root user, you need to | ||
# indicate a directory writeable by $NODEUSER to store the PID file | |||
# NB : 1) /usr/local/var/run does not exist by DEFAULT. Either create it | |||
# or choose one of your own liking. | |||
# 2) node, npm,... are best NOT installed/run as ROOT. | |||
# (see here: https://github.com/isaacs/npm/blob/master/README.md) | |||
NAME=node # name of the node.js executable | NAME=node # name of the node.js executable | ||
Строка 99: | Строка 99: | ||
# Exit if the 'run' folder is not present | # Exit if the 'run' folder is not present | ||
[ -d "$LOCAL_VAR_RUN" ] || { echo "Directory $LOCAL_VAR_RUN does not exist. Modify the '$INIT_SCRIPT_NAME_NOEXT' init.d | [ -d "$LOCAL_VAR_RUN" ] || { echo "Directory $LOCAL_VAR_RUN does not exist. Modify the '$INIT_SCRIPT_NAME_NOEXT' init.d script ($THIS_ARG) accordingly" >&2; exit 0; } | ||
# Read configuration variable file if it is present | # Read configuration variable file if it is present | ||
Строка 119: | Строка 119: | ||
do_start() | do_start() | ||
{ | { | ||
# Return | |||
# 0 if daemon has been started | |||
# 1 if daemon was already running | |||
# 2 if daemon could not be started | |||
start-stop-daemon --start --quiet --pidfile $PIDFILE --chuid $NODEUSER --background --exec $DAEMON --test > /dev/null \ | |||
|| { [ "$VERBOSE" != no ] && log_daemon_msg " ---> Daemon already running $DESC" "$INIT_SCRIPT_NAME_NOEXT"; return 1; } | |||
start-stop-daemon --start --quiet --chuid $NODEUSER --make-pidfile --pidfile $PIDFILE --background --exec $DAEMON -- \ | |||
$DAEMON_ARGS \ | |||
|| { [ "$VERBOSE" != no ] && log_daemon_msg " ---> could not be start $DESC" "$INIT_SCRIPT_NAME_NOEXT"; return 2; } | |||
# Add code here, if necessary, that waits for the process to be ready | |||
# to handle requests from services started subsequently which depend | |||
# on this one. As a last resort, sleep for some time. | |||
[ "$VERBOSE" != no ] && log_daemon_msg " ---> started $DESC" "$INIT_SCRIPT_NAME_NOEXT" | |||
} | } | ||
Строка 139: | Строка 139: | ||
do_stop() | do_stop() | ||
{ | { | ||
# Return | |||
# 0 if daemon has been stopped | |||
# 1 if daemon was already stopped | |||
# 2 if daemon could not be stopped | |||
# other if a failure occurred | |||
start-stop-daemon --stop --quiet --retry=TERM/30/KILL/5 --pidfile $PIDFILE --chuid $NODEUSER --name $DAEMON | |||
RETVAL="$?" | |||
#[ "$VERBOSE" != no ] && [ "$RETVAL" = 1 ] && log_daemon_msg " ---> SIGKILL failed => hardkill $DESC" "$INIT_SCRIPT_NAME_NOEXT" | |||
[ "$RETVAL" = 2 ] && return 2 | |||
# Wait for children to finish too if this is a daemon that forks | |||
# and if the daemon is only ever run from this initscript. | |||
# If the above conditions are not satisfied then add some other code | |||
# that waits for the process to drop all resources that could be | |||
# needed by services started subsequently. A last resort is to | |||
# sleep for some time. | |||
start-stop-daemon --stop --quiet --oknodo --retry=0/3/KILL/5 --pidfile $PIDFILE --chuid $NODEUSER --exec $DAEMON -- $DAEMON_ARGS | |||
[ "$?" = 2 ] && return 2 | |||
# Many daemons don't delete their pidfiles when they exit. | |||
rm -f $PIDFILE | |||
[ "$VERBOSE" != no ] && [ "$RETVAL" = 1 ] && log_daemon_msg " ---> $DESC not running" "$INIT_SCRIPT_NAME_NOEXT" | [ "$VERBOSE" != no ] && [ "$RETVAL" = 1 ] && log_daemon_msg " ---> $DESC not running" "$INIT_SCRIPT_NAME_NOEXT" | ||
[ "$VERBOSE" != no -a "$RETVAL" = 0 ] && log_daemon_msg " ---> $DESC stopped" "$INIT_SCRIPT_NAME_NOEXT" | [ "$VERBOSE" != no -a "$RETVAL" = 0 ] && log_daemon_msg " ---> $DESC stopped" "$INIT_SCRIPT_NAME_NOEXT" | ||
return "$RETVAL" | |||
} | } | ||
Строка 167: | Строка 167: | ||
# | # | ||
do_reload() { | do_reload() { | ||
# | |||
# If the daemon can reload its configuration without | |||
# restarting (for example, when it is sent a SIGHUP), | |||
# then implement that here. | |||
# | |||
start-stop-daemon --stop --quiet --signal 1 --pidfile $PIDFILE --chuid $NODEUSER --name $NAME | |||
return 0 | |||
} | } | ||
Строка 195: | Строка 195: | ||
if [ -n "$RUNNING" ]; then | if [ -n "$RUNNING" ]; then | ||
if [ $ISPIDACTIVE ]; then | if [ $ISPIDACTIVE ]; then | ||
log_success_msg "$INIT_SCRIPT_NAME_NOEXT (launched by $USER) (--chuid $NODEUSER) is running" | |||
exit 0 | |||
fi | fi | ||
else | else | ||
if [ -f $PIDFILE ]; then | if [ -f $PIDFILE ]; then | ||
log_success_msg "$INIT_SCRIPT_NAME_NOEXT (launched by $USER) (--chuid $NODEUSER) is not running, phantom pidfile $PIDFILE" | |||
exit 1 | |||
else | else | ||
log_success_msg "no instance launched by $USER, of $INIT_SCRIPT_NAME_NOEXT (--chuid $NODEUSER) found" | |||
exit 3 | |||
fi | fi | ||
fi | fi | ||
Строка 221: | Строка 221: | ||
case "$1" in | case "$1" in | ||
start) | start) | ||
[ "$VERBOSE" != no ] && log_daemon_msg "Starting $DESC" "$INIT_SCRIPT_NAME_NOEXT" | |||
do_start | |||
case "$?" in | |||
0|1) [ "$VERBOSE" != no ] && log_end_msg 0 ;; | |||
2) [ "$VERBOSE" != no ] && log_end_msg 1 ;; | |||
esac | |||
;; | |||
stop) | stop) | ||
[ "$VERBOSE" != no ] && log_daemon_msg "Stopping $DESC" "$INIT_SCRIPT_NAME_NOEXT" | |||
do_stop | |||
case "$?" in | |||
0|1) [ "$VERBOSE" != no ] && log_end_msg 0 ;; | |||
2) [ "$VERBOSE" != no ] && log_end_msg 1 ;; | |||
esac | |||
;; | |||
#reload|force-reload) | #reload|force-reload) | ||
# | |||
# If do_reload() is not implemented then leave this commented out | |||
# and leave 'force-reload' as an alias for 'restart'. | |||
# | |||
#log_daemon_msg "Reloading $DESC" "$NAME" | |||
#do_reload | |||
#log_end_msg $? | |||
#;; | |||
restart|force-reload) | restart|force-reload) | ||
# | |||
# If the "reload" option is implemented then remove the | |||
# 'force-reload' alias | |||
# | |||
log_daemon_msg "Restarting $DESC" "$INIT_SCRIPT_NAME_NOEXT" | |||
do_stop | |||
case "$?" in | |||
0|1) | |||
do_start | |||
case "$?" in | |||
0) log_end_msg 0 ;; | |||
1) log_end_msg 1 ;; # Old process is still running | |||
*) log_end_msg 1 ;; # Failed to start | |||
esac | |||
;; | |||
*) | |||
# Failed to stop | |||
log_end_msg 1 | |||
;; | |||
esac | |||
;; | |||
status) | status) | ||
do_status | do_status | ||
;; | ;; | ||
*) | *) | ||
#echo "Usage: $SCRIPTNAME {start|stop|restart|reload|force-reload}" >&2 | |||
echo "Usage: $SCRIPTNAME {start|stop|restart|force-reload}" >&2 | |||
exit 3 | |||
;; | |||
esac | esac | ||
exit 0 | exit 0 | ||
<hr> | |||
[https://gist.github.com/peterhost/715255 взято тут...] | [https://gist.github.com/peterhost/715255 взято тут...] | ||
==Если init скрипт не запускает приложение == | |||
То по-любому ошибки. Отловить ошибки помог другой init скрипт. В нем в скачестве демона не nodejs, а [https://ru.wikipedia.org/wiki/Nohup| nohup]. Очеень удобно при отладке. | |||
<nowiki> | |||
#! /bin/sh | |||
### BEGIN INIT INFO | |||
# Provides: mynodeapp-node-server | |||
# Required-Start: $local_fs $remote_fs $network $syslog | |||
# Required-Stop: $local_fs $remote_fs $network $syslog | |||
# Default-Start: 2 3 4 5 | |||
# Default-Stop: 0 1 6 | |||
# Short-Description: Starting mynodeapp node-server | |||
# Description: Start-stop-reload production mynodeapp node-server | |||
### END INIT INFO | |||
# Author: Name | |||
PATH=/sbin:/usr/sbin:/bin:/usr/bin | |||
DESC="app init" | |||
NAME=init.js | |||
DIR=/home/user | |||
ENGINE=$DIR/init.js | |||
DAEMON="/usr/bin/nohup" | |||
DAEMON_OPTS=" /usr/local/bin/node "$ENGINE" > /home/user/log/init.log" | |||
PIDFILE=/var/run/$NAME.pid | |||
test -x $DAEMON || exit 0 | |||
set -e | |||
. /lib/lsb/init-functions | |||
case "$1" in | |||
start) | |||
echo -n "Starting $DESC: " | |||
$DAEMON $DAEMON_OPTS & | |||
echo $! > $PIDFILE | |||
echo "$NAME." | |||
;; | |||
stop) | |||
echo -n "Stopping $DESC: " | |||
kill -9 $(cat $PIDFILE) | |||
rm $PIDFILE | |||
echo "$NAME." | |||
;; | |||
*) | |||
echo "Usage: $NAME {start|stop}" >&2 | |||
exit 1 | |||
;; | |||
esac</nowiki> |
Текущая версия от 12:52, 2 марта 2015
#!/bin/sh # ------------------------------------------------------------------------------ # SOME INFOS : fairly standard (debian) init script. # Note that node doesn't create a PID file (hence --make-pidfile) # has to be run in the background (hence --background) # and NOT as root (hence --chuid) # # MORE INFOS : INIT SCRIPT http://www.debian.org/doc/debian-policy/ch-opersys.html#s-sysvinit # INIT-INFO RULES http://wiki.debian.org/LSBInitScripts # INSTALL/REMOVE http://www.debian-administration.org/articles/28 # ------------------------------------------------------------------------------ # # # BEGIN <MODIFY TO REFLECT YOUR SETTINGS> # # # # # 1) Don't forget to also modify the COMMENTED fields in the "### BEGIN INIT INFO" # below (don't uncomment them) if you wish to install system-wide with update-rc.d # eg: provides, Short-Description, Description # # 2) in case you have different node.js servers running, each init.d script should # (obviously) have a UNIQUE BASE name so that PIDS do not conflict # --> name them accordingly # eg: node_static_server, node_express1, node_load_balancer.sh... # # 3) copy the renamed/modified script(s) to /etc/init.d # chmod 755, # # 4) if you wish the Daemon to be lauched at boot / stopped at shutdown : # INSTALL : update-rc.d scriptname defaults # (UNINSTALL : update-rc.d -f scriptname remove) # # 5) based on : Debian /etc/init.d/skeleton # modified by : Peter Host (www.oghme.com) # ______________________________________________________________________________ ### BEGIN INIT INFO # Provides: node_debian_init # Required-Start: $remote_fs $named $syslog # Required-Stop: $remote_fs $named $syslog # Default-Start: 2 3 4 5 # Default-Stop: 0 1 6 # Short-Description: DEBIAN initscript for node.js servers/apps # Description: ex : proxy server is a node.js http server listening on # port 8080 (relayed from 80 by iptables). It balances # http requests between the main nodejs server # (nodejs.mydomain.com:8000), the static file-server # (static.mydomain.com) and the legacy apache server # (apache.mydomain.com) and possibly other servers # place this file in /etc/init.d. ### END INIT INFO # Author: My Name <myname@domain.tld> # # Please remove the "Author" lines above and replace them # with your own name if you copy and modify this script. # ______________________________________________________________________________ # # PATH should only include /usr/* if it runs after the mountnfs.sh script PATH=/sbin:/usr/sbin:/bin:/usr/bin:/usr/local/bin # modify if you need DAEMON_ARGS="/path/to/app.js" # path to your node.js server/app # NB: don't use ~/ in path DESC="node.js http server" # whatever fancy description you like NODEUSER=myuser:mygroup # USER who OWNS the daemon process (no matter whoever runs the init script) # user:group (if no group is specified, the primary GID for that user is used) LOCAL_VAR_RUN=/usr/local/var/run # in case the init script is run by non-root user, you need to # indicate a directory writeable by $NODEUSER to store the PID file # NB : 1) /usr/local/var/run does not exist by DEFAULT. Either create it # or choose one of your own liking. # 2) node, npm,... are best NOT installed/run as ROOT. # (see here: https://github.com/isaacs/npm/blob/master/README.md) NAME=node # name of the node.js executable DAEMON=/usr/local/bin/$NAME # this SHOULD POINT TO where your node executable is # # # # END </MODIFY TO REFLECT YOUR SETTINGS> # # (Nothing else to modify from this point on...) # # ------------------------------------------------------------------------------ # Do NOT "set -e" [ $UID -eq "0" ] && LOCAL_VAR_RUN=/var/run # in case this script is run by root, override user setting THIS_ARG=$0 INIT_SCRIPT_NAME=`basename $THIS_ARG` [ -h $THIS_ARG ] && INIT_SCRIPT_NAME=`basename $(readlink $THIS_ARG)` # in case of symlink INIT_SCRIPT_NAME_NOEXT=${INIT_SCRIPT_NAME%.*} PIDFILE="$LOCAL_VAR_RUN/$INIT_SCRIPT_NAME_NOEXT.pid" SCRIPTNAME=/etc/init.d/$INIT_SCRIPT_NAME # Exit if the package is not installed [ -x "$DAEMON" ] || { echo "can't find Node.js ($DAEMON)" >&2; exit 0; } # Exit if the 'run' folder is not present [ -d "$LOCAL_VAR_RUN" ] || { echo "Directory $LOCAL_VAR_RUN does not exist. Modify the '$INIT_SCRIPT_NAME_NOEXT' init.d script ($THIS_ARG) accordingly" >&2; exit 0; } # Read configuration variable file if it is present [ -r /etc/default/$INIT_SCRIPT_NAME ] && . /etc/default/$INIT_SCRIPT_NAME # Load the VERBOSE setting and other rcS variables . /lib/init/vars.sh # Define LSB log_* functions. # Depend on lsb-base (>= 3.0-6) to ensure that this file is present. . /lib/lsb/init-functions # uncomment to override system setting # VERBOSE=yes # # Function that starts the daemon/service # do_start() { # Return # 0 if daemon has been started # 1 if daemon was already running # 2 if daemon could not be started start-stop-daemon --start --quiet --pidfile $PIDFILE --chuid $NODEUSER --background --exec $DAEMON --test > /dev/null \ || { [ "$VERBOSE" != no ] && log_daemon_msg " ---> Daemon already running $DESC" "$INIT_SCRIPT_NAME_NOEXT"; return 1; } start-stop-daemon --start --quiet --chuid $NODEUSER --make-pidfile --pidfile $PIDFILE --background --exec $DAEMON -- \ $DAEMON_ARGS \ || { [ "$VERBOSE" != no ] && log_daemon_msg " ---> could not be start $DESC" "$INIT_SCRIPT_NAME_NOEXT"; return 2; } # Add code here, if necessary, that waits for the process to be ready # to handle requests from services started subsequently which depend # on this one. As a last resort, sleep for some time. [ "$VERBOSE" != no ] && log_daemon_msg " ---> started $DESC" "$INIT_SCRIPT_NAME_NOEXT" } # # Function that stops the daemon/service # do_stop() { # Return # 0 if daemon has been stopped # 1 if daemon was already stopped # 2 if daemon could not be stopped # other if a failure occurred start-stop-daemon --stop --quiet --retry=TERM/30/KILL/5 --pidfile $PIDFILE --chuid $NODEUSER --name $DAEMON RETVAL="$?" #[ "$VERBOSE" != no ] && [ "$RETVAL" = 1 ] && log_daemon_msg " ---> SIGKILL failed => hardkill $DESC" "$INIT_SCRIPT_NAME_NOEXT" [ "$RETVAL" = 2 ] && return 2 # Wait for children to finish too if this is a daemon that forks # and if the daemon is only ever run from this initscript. # If the above conditions are not satisfied then add some other code # that waits for the process to drop all resources that could be # needed by services started subsequently. A last resort is to # sleep for some time. start-stop-daemon --stop --quiet --oknodo --retry=0/3/KILL/5 --pidfile $PIDFILE --chuid $NODEUSER --exec $DAEMON -- $DAEMON_ARGS [ "$?" = 2 ] && return 2 # Many daemons don't delete their pidfiles when they exit. rm -f $PIDFILE [ "$VERBOSE" != no ] && [ "$RETVAL" = 1 ] && log_daemon_msg " ---> $DESC not running" "$INIT_SCRIPT_NAME_NOEXT" [ "$VERBOSE" != no -a "$RETVAL" = 0 ] && log_daemon_msg " ---> $DESC stopped" "$INIT_SCRIPT_NAME_NOEXT" return "$RETVAL" } # # Function that sends a SIGHUP to the daemon/service # do_reload() { # # If the daemon can reload its configuration without # restarting (for example, when it is sent a SIGHUP), # then implement that here. # start-stop-daemon --stop --quiet --signal 1 --pidfile $PIDFILE --chuid $NODEUSER --name $NAME return 0 } # # Function that returns the daemon # do_status() { # # http://refspecs.freestandards.org/LSB_3.1.1/LSB-Core-generic/LSB-Core-generic/iniscrptact.html # 0 program is running or service is OK # 1 program is dead and /var/run pid file exists # (2 program is dead and /var/lock lock file exists) (not used here) # 3 program is not running # 4 program or service status is unknown RUNNING=$(running) # $PIDFILE corresponds to a live $NAME process ispidactive=$(pidof $NAME | grep `cat $PIDFILE 2>&1` >/dev/null 2>&1) ISPIDACTIVE=$? if [ -n "$RUNNING" ]; then if [ $ISPIDACTIVE ]; then log_success_msg "$INIT_SCRIPT_NAME_NOEXT (launched by $USER) (--chuid $NODEUSER) is running" exit 0 fi else if [ -f $PIDFILE ]; then log_success_msg "$INIT_SCRIPT_NAME_NOEXT (launched by $USER) (--chuid $NODEUSER) is not running, phantom pidfile $PIDFILE" exit 1 else log_success_msg "no instance launched by $USER, of $INIT_SCRIPT_NAME_NOEXT (--chuid $NODEUSER) found" exit 3 fi fi } running() { RUNSTAT=$(start-stop-daemon --start --quiet --pidfile $PIDFILE --chuid $NODEUSER --background --exec $DAEMON --test > /dev/null) if [ "$?" = 1 ]; then echo y fi } case "$1" in start) [ "$VERBOSE" != no ] && log_daemon_msg "Starting $DESC" "$INIT_SCRIPT_NAME_NOEXT" do_start case "$?" in 0|1) [ "$VERBOSE" != no ] && log_end_msg 0 ;; 2) [ "$VERBOSE" != no ] && log_end_msg 1 ;; esac ;; stop) [ "$VERBOSE" != no ] && log_daemon_msg "Stopping $DESC" "$INIT_SCRIPT_NAME_NOEXT" do_stop case "$?" in 0|1) [ "$VERBOSE" != no ] && log_end_msg 0 ;; 2) [ "$VERBOSE" != no ] && log_end_msg 1 ;; esac ;; #reload|force-reload) # # If do_reload() is not implemented then leave this commented out # and leave 'force-reload' as an alias for 'restart'. # #log_daemon_msg "Reloading $DESC" "$NAME" #do_reload #log_end_msg $? #;; restart|force-reload) # # If the "reload" option is implemented then remove the # 'force-reload' alias # log_daemon_msg "Restarting $DESC" "$INIT_SCRIPT_NAME_NOEXT" do_stop case "$?" in 0|1) do_start case "$?" in 0) log_end_msg 0 ;; 1) log_end_msg 1 ;; # Old process is still running *) log_end_msg 1 ;; # Failed to start esac ;; *) # Failed to stop log_end_msg 1 ;; esac ;; status) do_status ;; *) #echo "Usage: $SCRIPTNAME {start|stop|restart|reload|force-reload}" >&2 echo "Usage: $SCRIPTNAME {start|stop|restart|force-reload}" >&2 exit 3 ;; esac exit 0
Если init скрипт не запускает приложение
То по-любому ошибки. Отловить ошибки помог другой init скрипт. В нем в скачестве демона не nodejs, а nohup. Очеень удобно при отладке.
#! /bin/sh ### BEGIN INIT INFO # Provides: mynodeapp-node-server # Required-Start: $local_fs $remote_fs $network $syslog # Required-Stop: $local_fs $remote_fs $network $syslog # Default-Start: 2 3 4 5 # Default-Stop: 0 1 6 # Short-Description: Starting mynodeapp node-server # Description: Start-stop-reload production mynodeapp node-server ### END INIT INFO # Author: Name PATH=/sbin:/usr/sbin:/bin:/usr/bin DESC="app init" NAME=init.js DIR=/home/user ENGINE=$DIR/init.js DAEMON="/usr/bin/nohup" DAEMON_OPTS=" /usr/local/bin/node "$ENGINE" > /home/user/log/init.log" PIDFILE=/var/run/$NAME.pid test -x $DAEMON || exit 0 set -e . /lib/lsb/init-functions case "$1" in start) echo -n "Starting $DESC: " $DAEMON $DAEMON_OPTS & echo $! > $PIDFILE echo "$NAME." ;; stop) echo -n "Stopping $DESC: " kill -9 $(cat $PIDFILE) rm $PIDFILE echo "$NAME." ;; *) echo "Usage: $NAME {start|stop}" >&2 exit 1 ;; esac