当前位置:   article > 正文

python中的daemon守护进程实现方法_python 设置daemon进程

python 设置daemon进程
守护进程是生存期长的一种进程。它们独立于控制终端并且周期性的执行某种任务或等待处理某些发生的事件。他们常常在系统引导装入时启动,在系统关闭时终止。

守护进程的特性
1.在后台运行
2.与其运行前的环境隔离开来。这些环境包括未关闭的文件描述符、控制终端、会话和进程组、工作目录以及文件创建掩码等。这些环境通常是守护进程从执行它的父进程(特别是shell)中继承下来的。
3.启动方式特殊,它可以在系统启动时从启动脚本/etc/rc.d中启动,可以由inetd守护进程启动,可以由crond启动,还可以由用户终端(通常是shell)执行。
总之,除开这些特殊性以外,守护进程与普通进程基本上没有什么区别。因此,编写守护进程实际上是把一个普通进程按照上述的守护进程的特性改造成为守护进程。

守护进程编程规则
1.在后台运行,调用fork ,然后使父进程exit
2.脱离控制终端,登录会话和进程组,调用setsid()使进程成为会话组长
3.禁止进程重新打开控制终端
4.关闭打开的文件描述符,调用fclose()
5.将当前工作目录更改为根目录。
6.重设文件创建掩码为0
7.处理SIGCHLD 信号

下面是一个的demo源码示例:

  1. #!/usr/bin/env python
  2. #encoding: utf-8
  3. #description: 一个守护进程的简单包装类, 具备常用的start|stop|restart|status功能, 使用方便
  4. # 需要改造为守护进程的程序只需要重写基类的run函数就可以了
  5. #date: 2015-10-29
  6. #usage: 启动: python daemon_class.py start
  7. # 关闭: python daemon_class.py stop
  8. # 状态: python daemon_class.py status
  9. # 重启: python daemon_class.py restart
  10. # 查看: ps -axj | grep daemon_class
  11. import atexit, os, sys, time, signal
  12. class CDaemon:
  13. '''
  14. a generic daemon class.
  15. usage: subclass the CDaemon class and override the run() method
  16. stderr 表示错误日志文件绝对路径, 收集启动过程中的错误日志
  17. verbose 表示将启动运行过程中的异常错误信息打印到终端,便于调试,建议非调试模式下关闭, 默认为1, 表示开启
  18. save_path 表示守护进程pid文件的绝对路径
  19. '''
  20. def __init__(self, save_path, stdin=os.devnull, stdout=os.devnull, stderr=os.devnull, home_dir='.', umask=022, verbose=1):
  21. self.stdin = stdin
  22. self.stdout = stdout
  23. self.stderr = stderr
  24. self.pidfile = save_path #pid文件绝对路径
  25. self.home_dir = home_dir
  26. self.verbose = verbose #调试开关
  27. self.umask = umask
  28. self.daemon_alive = True
  29. def daemonize(self):
  30. try:
  31. pid = os.fork()
  32. if pid > 0:
  33. sys.exit(0)
  34. except OSError, e:
  35. sys.stderr.write('fork #1 failed: %d (%s)\n' % (e.errno, e.strerror))
  36. sys.exit(1)
  37. os.chdir(self.home_dir)
  38. os.setsid()
  39. os.umask(self.umask)
  40. try:
  41. pid = os.fork()
  42. if pid > 0:
  43. sys.exit(0)
  44. except OSError, e:
  45. sys.stderr.write('fork #2 failed: %d (%s)\n' % (e.errno, e.strerror))
  46. sys.exit(1)
  47. sys.stdout.flush()
  48. sys.stderr.flush()
  49. si = file(self.stdin, 'r')
  50. so = file(self.stdout, 'a+')
  51. if self.stderr:
  52. se = file(self.stderr, 'a+', 0)
  53. else:
  54. se = so
  55. os.dup2(si.fileno(), sys.stdin.fileno())
  56. os.dup2(so.fileno(), sys.stdout.fileno())
  57. os.dup2(se.fileno(), sys.stderr.fileno())
  58. def sig_handler(signum, frame):
  59. self.daemon_alive = False
  60. signal.signal(signal.SIGTERM, sig_handler)
  61. signal.signal(signal.SIGINT, sig_handler)
  62. if self.verbose >= 1:
  63. print 'daemon process started ...'
  64. atexit.register(self.del_pid)
  65. pid = str(os.getpid())
  66. file(self.pidfile, 'w+').write('%s\n' % pid)
  67. def get_pid(self):
  68. try:
  69. pf = file(self.pidfile, 'r')
  70. pid = int(pf.read().strip())
  71. pf.close()
  72. except IOError:
  73. pid = None
  74. except SystemExit:
  75. pid = None
  76. return pid
  77. def del_pid(self):
  78. if os.path.exists(self.pidfile):
  79. os.remove(self.pidfile)
  80. def start(self, *args, **kwargs):
  81. if self.verbose >= 1:
  82. print 'ready to starting ......'
  83. #check for a pid file to see if the daemon already runs
  84. pid = self.get_pid()
  85. if pid:
  86. msg = 'pid file %s already exists, is it already running?\n'
  87. sys.stderr.write(msg % self.pidfile)
  88. sys.exit(1)
  89. #start the daemon
  90. self.daemonize()
  91. self.run(*args, **kwargs)
  92. def stop(self):
  93. if self.verbose >= 1:
  94. print 'stopping ...'
  95. pid = self.get_pid()
  96. if not pid:
  97. msg = 'pid file [%s] does not exist. Not running?\n' % self.pidfile
  98. sys.stderr.write(msg)
  99. if os.path.exists(self.pidfile):
  100. os.remove(self.pidfile)
  101. return
  102. #try to kill the daemon process
  103. try:
  104. i = 0
  105. while 1:
  106. os.kill(pid, signal.SIGTERM)
  107. time.sleep(0.1)
  108. i = i + 1
  109. if i % 10 == 0:
  110. os.kill(pid, signal.SIGHUP)
  111. except OSError, err:
  112. err = str(err)
  113. if err.find('No such process') > 0:
  114. if os.path.exists(self.pidfile):
  115. os.remove(self.pidfile)
  116. else:
  117. print str(err)
  118. sys.exit(1)
  119. if self.verbose >= 1:
  120. print 'Stopped!'
  121. def restart(self, *args, **kwargs):
  122. self.stop()
  123. self.start(*args, **kwargs)
  124. def is_running(self):
  125. pid = self.get_pid()
  126. #print(pid)
  127. return pid and os.path.exists('/proc/%d' % pid)
  128. def run(self, *args, **kwargs):
  129. 'NOTE: override the method in subclass'
  130. print 'base class run()'
  131. class ClientDaemon(CDaemon):
  132. def __init__(self, name, save_path, stdin=os.devnull, stdout=os.devnull, stderr=os.devnull, home_dir='.', umask=022, verbose=1):
  133. CDaemon.__init__(self, save_path, stdin, stdout, stderr, home_dir, umask, verbose)
  134. self.name = name #派生守护进程类的名称
  135. def run(self, output_fn, **kwargs):
  136. fd = open(output_fn, 'w')
  137. while True:
  138. line = time.ctime() + '\n'
  139. fd.write(line)
  140. fd.flush()
  141. time.sleep(1)
  142. fd.close()
  143. if __name__ == '__main__':
  144. help_msg = 'Usage: python %s <start|stop|restart|status>' % sys.argv[0]
  145. if len(sys.argv) != 2:
  146. print help_msg
  147. sys.exit(1)
  148. p_name = 'clientd' #守护进程名称
  149. pid_fn = '/tmp/daemon_class.pid' #守护进程pid文件的绝对路径
  150. log_fn = '/tmp/daemon_class.log' #守护进程日志文件的绝对路径
  151. err_fn = '/tmp/daemon_class.err.log' #守护进程启动过程中的错误日志,内部出错能从这里看到
  152. cD = ClientDaemon(p_name, pid_fn, stderr=err_fn, verbose=1)
  153. if sys.argv[1] == 'start':
  154. cD.start(log_fn)
  155. elif sys.argv[1] == 'stop':
  156. cD.stop()
  157. elif sys.argv[1] == 'restart':
  158. cD.restart(log_fn)
  159. elif sys.argv[1] == 'status':
  160. alive = cD.is_running()
  161. if alive:
  162. print 'process [%s] is running ......' % cD.get_pid()
  163. else:
  164. print 'daemon process [%s] stopped' %cD.name
  165. else:
  166. print 'invalid argument!'
  167. print help_msg
下面是运行截图

产生的日志文件为

参考文档
http://www.jb51.net/article/54199.htm 都不错,这个守护进程类包装非常完备,我已经重新整理了一遍

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/236244
推荐阅读
相关标签
  

闽ICP备14008679号