Демон Python не будет работать в фоновом режиме в Ubuntu - PullRequest
5 голосов
/ 10 августа 2010

Мой демон Python отлично работает на переднем плане моей системы Ubuntu, используя эту команду в терминале:

python /opt/my-daemon.py foreground

Однако, когда я пытаюсь вызвать демон, используя "start" команда терпит неудачу, почему?

python /opt/my-daemon.py start

Вот как я называю команду в файле /etc/rc.local:

python /opt/my-daemon.py start &

При этом код:

1.daemon.py

#!/usr/bin/env python
import sys, os, time, atexit
from signal import SIGTERM
class Daemon:
"""
A generic daemon class.

Usage: subclass the Daemon class and override the run() method
"""
def __init__(self, pidfile,
    stdin='/dev/null',stdout='/dev/null',stderr='/dev/null'):
    self.stdin = stdin
    self.stdout = stdout
    self.stderr = stderr
    self.pidfile = pidfile

def daemonize(self):
    """
    Do the UNIX double-fork magic. See Richard Stevens' "Advanced
    Programming in the UNIX Environment" for details (ISBN 0201563177)
    http://www.erlenstar.demon.co.uk/unix/faq_2.html#SEC16
    """
    try:
        pid = os.fork()
        if pid > 0:
            # exit first parent
            sys.exit(0)
    except OSError, e:
        sys.stderr.write("fork #1 failed: %d (%s)\n" % (e.errno,
                    e.strerror))
        sys.exit(1)
    # Decouple from parent environment
    os.chdir("/")
    os.setsid()
    os.umask(0)

    # Do second fork
    try:
        pid = os.fork()
        if pid > 0:
            # Exit from second parent
            sys.exit(0)
    except OSError, e:
        sys.stderr.write("fork #2 failed: %d (%s)\n" % (e.errno, e.strerror))
        sys.exit(1)

    # Redirect standard file descriptors
    sys.stdout.flush()
    sys.stderr.flush()
    si = file(self.stdin, 'r')
    so = file(self.stdout, 'a+')
    se = file(self.stderr, 'a+', 0)
    os.dup2(si.fileno(), sys.stdin.fileno())
    os.dup2(so.fileno(), sys.stdout.fileno())
    os.dup2(se.fileno(), sys.stderr.fileno())

    # Write pidfile
    atexit.register(self.delpid)
    pid = str(os.getpid())
    file(self.pidfile,'w+').write("%s\n" % pid)

def delpid(self):
    os.remove(self.pidfile)

def start(self):
    """
    Start the daemon
    """
    # Check for a pidfile to see if the daemon already runs
    try:
        pf = file(self.pidfile,'r')
        pid = int(pf.read().strip())
        pf.close()
    except IOError:
        pid = None

    if pid:
        message = "pidfile %s already exist. Daemon already running?\n"
        sys.stderr.write(message % self.pidfile)
        sys.exit(1)

    # Start the daemon
    self.daemonize()
    self.run()

def stop(self):
    """
    Stop the daemon
    """
    # Get the pid from the pidfile
    try:
        pf = file(self.pidfile,'r')
        pid = int(pf.read().strip())
        pf.close()
    except IOError:
        pid = None

    if not pid:
        message = "pidfile %s does not exist. Daemon not running?\n"
        sys.stderr.write(message % self.pidfile)
        return # not an error in a restart

    # Try killing the daemon process
    try:
        while 1:
            os.kill(pid, SIGTERM)
            time.sleep(0.1)
    except OSError, err:
        err = str(err)
        if err.find("No such process") > 0:
            if os.path.exists(self.pidfile):
                os.remove(self.pidfile)
        else:
            print str(err)
            sys.exit(1)

def restart(self):
    """
    Restart the daemon
    """
    self.stop()
    self.start()

def run(self):
    """
    You should override this method when you subclass Daemon. It will be called after the process has been
    daemonized by start() or restart().
    """

2.my-daemon.py

import sys, time
from daemon import Daemon
import MySQLdb #MySQL libraries
#Database parameters
config = {"host":"localhost",...}
try:
    conn = MySQLdb.connect(config['host'],...
class MyDaemon(Daemon):
def run(self):
    while True:
        time.sleep(2)
                    #{Do processes, connect to the database, etc....}
                    ...
if __name__ == "__main__":
daemon = MyDaemon('/tmp/daemon-example.pid')
if len(sys.argv) == 2:
    if 'start' == sys.argv[1]:
        daemon.start()
    elif 'stop' == sys.argv[1]:
        daemon.stop()
    elif 'restart' == sys.argv[1]:
        daemon.restart()
    elif 'foreground' == sys.argv[1]: #This runs the daemon in the foreground
        daemon.run()
    else:
        print "Unknown command"
        sys.exit(2)
    sys.exit(0)
else:
    print "usage: %s start|foreground|stop|restart" % sys.argv[0]
    sys.exit(2)

Ответы [ 3 ]

1 голос
/ 10 августа 2010

решаемые .У меня сложилось впечатление, что параметры foreground и start - это две разные вещи.Оказывается, мне просто нужно было сделать следующее.

def run(self):
    while True:
        time.sleep(2)

до

def start(self):
    while True:
        time.sleep(2)

Затем я удалил параметр foreground, потому что я могу запустить скрипт из терминала, используяstart команда для просмотра вывода на переднем плане.

python /opt/my-daemon.py start

Кроме того, в rc.local я запускаю скрипт следующим образом:

python /opt/my-daemon.py start &

Это скрывает процесс демона и выполняетскрипт при запуске независимо от того, кто входит в систему:)

0 голосов
/ 21 сентября 2010
0 голосов
/ 10 августа 2010

Вместо того, чтобы использовать daemon.py, вы можете рассмотреть возможность использования Upstart системы Ubuntu , которая предоставляет простой способ настройки респавна. По той же ссылке, он показывает:

* Services may be respawned if they die unexpectedly
* Supervision and respawning of daemons which separate from their parent process

Если вы используете Ubuntu9.10 или новее, взгляните на /etc/init/cron.conf в качестве примера. Я полагаю, что для более ранних версий Ubuntu сценарии запуска находятся в /etc/event.d/.

.

Объяснение ключевых слов Upstart см. здесь .

...