[Zope-Checkins] CVS: Zope/lib/python/zdaemon - Daemon.py:1.10
Chris McDonough
chrism@zope.com
Sat, 15 Jun 2002 00:19:23 -0400
Update of /cvs-repository/Zope/lib/python/zdaemon
In directory cvs.zope.org:/tmp/cvs-serv2819
Modified Files:
Daemon.py
Log Message:
Cleaned up Daemon:
- Removed unused exceptions.
- Refactored fork code and mainloop.
- Removed dependencies on Heartbeat. Nobody uses this.
- Introduced SignalPasser.
- Hardcoded interesting signals instead of relying on them being passed in.
=== Zope/lib/python/zdaemon/Daemon.py 1.9 => 1.10 ===
##############################################################################
-import os, sys, time, posix, signal
+import os, sys, time, signal
from ZDaemonLogging import pstamp
-import Heartbeat
import zLOG
+from SignalPasser import SignalPasser
pyth = sys.executable
-class KidDiedOnMeError(Exception):
+class DieNow(Exception):
pass
-class ExecError(Exception):
- pass
-
-class ForkError(Exception):
- pass
+def run(argv, pidfile=''):
+ if os.environ.has_key('ZDAEMON_MANAGED'):
+ # We're being run by the child.
+ return
-FORK_ATTEMPTS = 2
+ os.environ['ZDAEMON_MANAGED']='TRUE'
-def forkit(attempts = FORK_ATTEMPTS):
- while attempts:
- # if at first you don't succeed...
- attempts = attempts - 1
+ if not os.environ.has_key('Z_DEBUG_MODE'):
+ detach() # detach from the controlling terminal
+
+ while 1:
try:
pid = os.fork()
- except os.error:
- pstamp('Houston, the fork failed', zLOG.ERROR)
- time.sleep(2)
+ if pid:
+ # We're the parent (the daemon process)
+ # pass all "normal" signals along to our child, but don't
+ # respond to them ourselves unless they say "die"!
+ interesting = [1, 2, 3, 10, 12, 15]
+ # ie. HUP, INT, QUIT, USR1, USR2, TERM
+ for sig in interesting:
+ signal.signal(sig, SignalPasser(sig))
+ pstamp('Houston, we have forked: pid %s' % pid, zLOG.INFO)
+ write_pidfile(pidfile)
+ p,s = wait(pid) # waitpid will block until child exit
+ if s:
+ # continue and restart because our child died
+ # with a nonzero exit code, meaning he bit it in
+ # an unsavory way (likely a segfault or something)
+ log_pid(p, s)
+ continue
+ else:
+ # no need to restart, our child wanted to die.
+ raise DieNow
+
+ else:
+ # we're the child (Zope/ZEO)
+ args = [pyth]
+ if not __debug__:
+ # we're running in optimized mode
+ args.append('-O')
+ os.execv(pyth, tuple(args) + tuple(argv))
+
+ except DieNow:
+ sys.exit()
+
+def detach():
+ # do the funky chicken dance to detach from the terminal
+ pid = os.fork()
+ if pid: sys.exit(0)
+ os.close(0); sys.stdin = open('/dev/null')
+ os.close(1); sys.stdout = open('/dev/null','w')
+ os.close(2); sys.stderr = open('/dev/null','w')
+ os.setsid()
+
+def write_pidfile(pidfile):
+ if pidfile:
+ pf = open(pidfile, 'w+')
+ pf.write(("%s" % os.getpid()))
+ pf.close()
+
+def wait(pid):
+ while 1:
+ 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:
- pstamp('Houston, we have forked', zLOG.INFO)
- return pid
+ return p, s
def log_pid(p, s):
if os.WIFEXITED(s):
@@ -72,99 +123,6 @@
signum)
pstamp('Aiieee! Process %s %s' % (p, msg),
zLOG.ERROR)
-
-def run(argv, pidfile='', signals=None):
- if signals is None:
- signals = []
- if os.environ.has_key('ZDAEMON_MANAGED'):
- # We're the child at this point.
- return
-
- os.environ['ZDAEMON_MANAGED']='TRUE'
-
- if not os.environ.has_key('Z_DEBUG_MODE'):
- # Detach from terminal
- pid = os.fork()
- if pid:
- sys.exit(0)
- os.close(0); sys.stdin = open('/dev/null')
- os.close(1); sys.stdout = open('/dev/null','w')
- os.close(2); sys.stderr = open('/dev/null','w')
- os.setsid()
-
- while 1:
-
- try:
- pid = forkit()
-
- if pid is None:
- raise ForkError
-
- elif pid:
- # 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:
- pf = open(pidfile, 'w+')
- pf.write(("%s" % os.getpid()))
- pf.close()
-
- while 1:
- 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:
- log_pid(p, s)
- else:
- pstamp(('The kid, %s, died on me.' % pid),
- zLOG.WARNING)
- raise ForkError
-
- raise KidDiedOnMeError
-
- else:
- # Child
- if __debug__:
- # non optimized
- os.execv(pyth, (pyth,) + tuple(argv))
- else:
- # optimized
- os.execv(pyth, (pyth, '-O') + tuple(argv))
-
- except ExecError:
- sys.exit()
- except ForkError:
- sys.exit()
- except KidDiedOnMeError:
- pass
_signals = None