2006-06-20 13:24:19 +08:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
|
|
|
|
if os.name == 'posix':
|
2008-06-30 19:22:41 +08:00
|
|
|
def become_daemon(our_home_dir='.', out_log='/dev/null',
|
|
|
|
err_log='/dev/null', umask=022):
|
2006-06-20 13:24:19 +08:00
|
|
|
"Robustly turn into a UNIX daemon, running in our_home_dir."
|
|
|
|
# First fork
|
|
|
|
try:
|
|
|
|
if os.fork() > 0:
|
|
|
|
sys.exit(0) # kill off parent
|
2012-04-29 00:09:37 +08:00
|
|
|
except OSError as e:
|
2006-06-20 13:24:19 +08:00
|
|
|
sys.stderr.write("fork #1 failed: (%d) %s\n" % (e.errno, e.strerror))
|
|
|
|
sys.exit(1)
|
|
|
|
os.setsid()
|
|
|
|
os.chdir(our_home_dir)
|
2008-06-30 19:22:41 +08:00
|
|
|
os.umask(umask)
|
2006-06-20 13:24:19 +08:00
|
|
|
|
|
|
|
# Second fork
|
|
|
|
try:
|
|
|
|
if os.fork() > 0:
|
2008-03-18 22:08:56 +08:00
|
|
|
os._exit(0)
|
2012-04-29 00:09:37 +08:00
|
|
|
except OSError as e:
|
2006-06-20 13:24:19 +08:00
|
|
|
sys.stderr.write("fork #2 failed: (%d) %s\n" % (e.errno, e.strerror))
|
2008-03-18 22:08:56 +08:00
|
|
|
os._exit(1)
|
2006-06-20 13:24:19 +08:00
|
|
|
|
|
|
|
si = open('/dev/null', 'r')
|
|
|
|
so = open(out_log, 'a+', 0)
|
|
|
|
se = open(err_log, 'a+', 0)
|
|
|
|
os.dup2(si.fileno(), sys.stdin.fileno())
|
|
|
|
os.dup2(so.fileno(), sys.stdout.fileno())
|
|
|
|
os.dup2(se.fileno(), sys.stderr.fileno())
|
2008-03-18 23:53:46 +08:00
|
|
|
# Set custom file descriptors so that they get proper buffering.
|
|
|
|
sys.stdout, sys.stderr = so, se
|
2006-06-20 13:24:19 +08:00
|
|
|
else:
|
2008-06-30 19:22:41 +08:00
|
|
|
def become_daemon(our_home_dir='.', out_log=None, err_log=None, umask=022):
|
2006-06-20 13:24:19 +08:00
|
|
|
"""
|
|
|
|
If we're not running under a POSIX system, just simulate the daemon
|
|
|
|
mode by doing redirections and directory changing.
|
|
|
|
"""
|
|
|
|
os.chdir(our_home_dir)
|
2008-06-30 19:22:41 +08:00
|
|
|
os.umask(umask)
|
2006-06-20 13:24:19 +08:00
|
|
|
sys.stdin.close()
|
|
|
|
sys.stdout.close()
|
|
|
|
sys.stderr.close()
|
|
|
|
if err_log:
|
|
|
|
sys.stderr = open(err_log, 'a', 0)
|
|
|
|
else:
|
|
|
|
sys.stderr = NullDevice()
|
|
|
|
if out_log:
|
|
|
|
sys.stdout = open(out_log, 'a', 0)
|
|
|
|
else:
|
|
|
|
sys.stdout = NullDevice()
|
|
|
|
|
|
|
|
class NullDevice:
|
|
|
|
"A writeable object that writes to nowhere -- like /dev/null."
|
|
|
|
def write(self, s):
|
|
|
|
pass
|