[Zope-Checkins] CVS: Zope/lib/python/zdaemon - SignalPasser.py:1.1.2.1 Daemon.py:1.2.2.1

Chris McDonough chrism@zope.com
Fri, 15 Mar 2002 14:36:03 -0500


Update of /cvs-repository/Zope/lib/python/zdaemon
In directory cvs.zope.org:/tmp/cvs-serv19763/lib/python/zdaemon

Modified Files:
      Tag: chrism_logrotate_branch
	Daemon.py 
Added Files:
      Tag: chrism_logrotate_branch
	SignalPasser.py 
Log Message:
Patched up z2.py and some of asyncore to be able to deal with signals.

In this branch, Zope will:

- open and close all of its logfiles upon receving a USR2 signal

- restart "cleanly" upon receiving a HUP signal

- shutdown "cleanly" upon receiving a TERM or INT signal.



=== Added File Zope/lib/python/zdaemon/SignalPasser.py ===
class SignalPasser:
    def __init__(self, pid):
        self.pid = pid

    def __call__(self, signum, frame):
        import os, sys, signal
        os.kill(self.pid, signum)
        if signum in [signal.SIGTERM, signal.SIGINT]:
            sys.exit(0)

def pass_signals_to_process(pid, signals):
    import signal
    for s in signals:
        signal.signal(s, SignalPasser(pid))


=== Zope/lib/python/zdaemon/Daemon.py 1.2 => 1.2.2.1 ===
 ##############################################################################
 
-import os, sys, time, signal, posix
+import os, sys, time, posix, signal
+ALL_SIGNALS = ['SIGABRT', 'SIGALRM', 'SIGBUS', 'SIGCHLD', 'SIGCLD', 'SIGCONT', 'SIGFPE', 'SIGHUP', 'SIGILL', 'SIGINT', 'SIGIO', 'SIGIOT', 'SIGKILL', 'SIGPIPE', 'SIGPOLL', 'SIGPROF', 'SIGPWR', 'SIGQUIT', 'SIGSEGV', 'SIGSTOP', 'SIGSYS', 'SIGTERM', 'SIGTRAP', 'SIGTSTP', 'SIGTTIN', 'SIGTTOU', 'SIGURG', 'SIGUSR1', 'SIGUSR2', 'SIGVTALRM', 'SIGWINCH', 'SIGXCPU', 'SIGXFSZ']
+
 
 from ZDaemonLogging import pstamp
 import Heartbeat
@@ -43,7 +45,9 @@
             pstamp('Houston, we have forked', zLOG.INFO)
             return pid
 
-def run(argv, pidfile=''):
+import signal
+
+def run(argv, pidfile='', signals=[]):
     if os.environ.has_key('ZDAEMON_MANAGED'):
         # We're the child at this point.
         return
@@ -66,7 +70,18 @@
                 raise ForkError
 
             elif pid:
-                # Parent 
+                # the process we're daemoning for can signify that it
+                # wants us to notify it when we get specific signals
+                #
+                #
+                # we always register TERM and INT so we can reap our child.
+                signals = signals + [signal.SIGTERM, signal.SIGINT]
+                # TERM happens on normal kill
+                # INT happens on Ctrl-C (debug mode)
+                import SignalPasser
+                SignalPasser.pass_signals_to_process(pid, signals)
+
+                # Parent
                 pstamp(('Hi, I just forked off a kid: %s' % pid), zLOG.INFO)
                 # here we want the pid of the parent
                 if pidfile:
@@ -74,24 +89,44 @@
                     pf.write(("%s" % os.getpid()))
                     pf.close()
 
-                while 1: 
-                    if not Heartbeat.BEAT_DELAY:
-                        p,s = os.waitpid(pid, 0)
-                    else:
-                        p,s = os.waitpid(pid, os.WNOHANG)
-                        if not p:
-                            time.sleep(Heartbeat.BEAT_DELAY)
-                            Heartbeat.heartbeat()
-                            continue
-                    if s:
-                        pstamp(('Aiieee! %s exited with error code: %s' 
-                                % (p, s)), zLOG.ERROR)
-                    else:
-                        pstamp(('The kid, %s, died on me.' % pid),
-                               zLOG.WARNING)
-                        raise ForkError
-
-                    raise KidDiedOnMeError
+                while 1:
+                    try:
+                        if not Heartbeat.BEAT_DELAY:
+                            try:
+                                p,s = os.waitpid(pid, 0)
+                            except OSError:
+                                # catch EINTR, it's raised as a result of
+                                # interrupting waitpid with a signal
+                                # and we don't care about it.
+                                continue
+                        else:
+                            try:
+                                p,s = os.waitpid(pid, os.WNOHANG)
+                            except OSError:
+                                # catch EINTR, it's raised as a result of
+                                # interrupting waitpid with a signal
+                                # and we don't care about it.
+                                p, s = None, None
+                            if not p:
+                                time.sleep(Heartbeat.BEAT_DELAY)
+                                Heartbeat.heartbeat()
+                                continue
+                        if s:
+                            pstamp(('Aiieee! %s exited with error code: %s' 
+                                    % (p, s)), zLOG.ERROR)
+                        else:
+                            pstamp(('The kid, %s, died on me.' % pid),
+                                   zLOG.WARNING)
+                            raise ForkError
+
+                        raise KidDiedOnMeError
+                    except OSError, num:
+                        if num == 4:
+                            # This is likely an interrupted system call
+                            # inside os.waitpid caused by a signal.
+                            # Though we have to catch it, we don't really
+                            # care about it and we just pass.
+                            pass
 
             else:
                 # Child