Migrar o script socat init para systemd

8

Eu uso o socat com o seguinte script init no debian 7.2 com sysVinit. Funciona perfeitamente:

#!/bin/bash
DESC=socat
DAEMON=/usr/bin/socat
LIB=/usr/lib/socat
SOCAT_ARGS="-d -d -lf /var/log/socat.log"

[ ! -f /etc/default/socat.conf ] || . /etc/default/socat.conf

. /lib/lsb/init-functions

PATH=/bin:/usr/bin:/sbin:/usr/sbin

[ -x $DAEMON ] || exit 0

#
#       Try to increase the # of filedescriptors we can open.
#
maxfds () {
        [ -n "$SOCAT_MAXFD" ] || return
        [ -f /proc/sys/fs/file-max ] || return 0
        [ $SOCAT_MAXFD -le 4096 ] || SQUID_MAXFD=4096
        global_file_max=`cat /proc/sys/fs/file-max`
        minimal_file_max=$(($SOCAT_MAXFD + 4096))
        if [ "$global_file_max" -lt $minimal_file_max ]
        then
                echo $minimal_file_max > /proc/sys/fs/file-max
        fi
        ulimit -n $SOCAT_MAXFD
}

start_socat() {
        start-stop-daemon --quiet --start \
                --pidfile /var/run/socat.$NAME.pid \
                --background --make-pidfile \
                --exec $DAEMON -- $SOCAT_ARGS $ARGS < /dev/null
}

stop_socat() {
        start-stop-daemon --stop --quiet --pidfile /var/run/socat.$NAME.pid --exec $DAEMON
        rm -f /var/run/socat.$NAME.pid
}

start () {
        echo "Starting $DESC:"

        maxfds
        umask 027
        cd /tmp
        if test "x$AUTOSTART" = "xnone" -o -z "x$AUTOSTART" ; then
                echo "Autostart disabled."
                exit 0
        fi
        for NAME in $AUTOSTART ; do
                ARGS=`eval echo \\\$SOCAT_$NAME`
                echo $ARGS
                start_socat
                echo " $NAME $ARGS"
        done
        return $?
}

stop () {
        echo -n "Stopping $DESC:"

        for PIDFILE in `ls /var/run/socat.*.pid 2> /dev/null`; do
                NAME=`echo $PIDFILE | cut -c16-`
                NAME=${NAME%%.pid}
                stop_socat
                echo -n " $NAME"
        done
}

case "$1" in
    start)
        log_daemon_msg "Starting socat" "socat"
        if start ; then
                log_end_msg $?
        else
                log_end_msg $?
        fi
        ;;
    stop)
        log_daemon_msg "Stopping socat" "socat"
        if stop ; then
                log_end_msg $?
        else
                log_end_msg $?
        fi
        ;;
    reload|force-reload|restart)
        log_daemon_msg "Restarting socat" "socat"
        stop
        if start ; then
                log_end_msg $?
        else
                log_end_msg $?
        fi
        ;;
        *)
        echo "Usage: /etc/init.d/$NAME {start|stop|reload|force-reload|restart}"
        exit 3
        ;;
esac

exit 0

No entanto, após uma atualização para o debian 7.4, o sistema mudou para systemd. Portanto, para executar o mesmo script no systemd, adicionei um serviço que envolve o script /etc/init.d/socat:

[Unit]
Description=Socat

[Service]
ExecStart=/etc/init.d/socat start
ExecStop=/etc/init.d/socat stop

[Install]
WantedBy=multi-user.target

Quando inicio o serviço, ele é iniciado, mas para diretamente:

Carregado: carregado (/usr/lib/systemd/system/socat.service; ativado)
Ativo: inativo (inativo) desde sexta, 18 de abril de 2014 14:09:46 +0200; 4s atrás Processo: 5334 ExecStart = / etc / init.d / socat start (código = encerrado, status = 0 / SUCCESS) CGroup: name = systemd: /system/socat.service

Estou esquecendo de algo?

Perdido em OWL
fonte
Estou esquecendo de algo? Sim, você perdeu realmente migrando o script init.d para systemd :)
Piotr Dobrogost

Respostas:

8

Acabei de descobrir que eu tenho que usar

Type=forking

como descrito em http://www.freedesktop.org/software/systemd/man/systemd.service.html .

Se definido como bifurcação, espera-se que o processo configurado com ExecStart = chame fork () como parte de sua inicialização. Espera-se que o processo pai saia quando a inicialização estiver concluída e todos os canais de comunicação estiverem configurados. A criança continua sendo executada como o processo principal daemon. Esse é o comportamento dos daemons tradicionais do UNIX. Se essa configuração for usada, é recomendável usar também a opção PIDFile =, para que o systemd possa identificar o processo principal do daemon. O systemd continuará com as unidades de acompanhamento assim que o processo pai terminar.

Perdido em OWL
fonte
1

Para a sociedade , eu uso uma abordagem sistemática pura . Este é um exemplo para um loopback serial:

[Unit]
Description=Socat Serial Loopback
#Before=my-other.service

[Service]
Type=simple
StandardOutput=syslog
StandardError=syslog
SyslogIdentifier=socat-serial-lo

ExecStart=/usr/bin/socat -d -d pty,raw,echo=0,link=/tmp/seriallo-a pty,raw,echo=0,link=/tmp/seriallo-b
Restart=always

[Install]
WantedBy=multi-user.target

Isso pode ser gravado em /etc/systemd/system/socat-serial-lo.service(no Ubuntu 16.04+) e, em seguida:

systemctl daemon-reload
systemctl start socat-serial-lo
systemctl enable socat-serial-lo  # (to start it during bootup)

Uma vantagem deste método é que a linha de comando definida por ExecStartpode ser testada diretamente da linha de comando sem alterações, para testar o comando.

jjmontes
fonte