platform.py 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. """celery.platform"""
  2. import os
  3. import sys
  4. import errno
  5. import resource
  6. # File mode creation mask of the daemon.
  7. # No point in changing this, as we don't really create any files.
  8. DAEMON_UMASK = 0
  9. # Default working directory for the daemon.
  10. DAEMON_WORKDIR = "/"
  11. # Default maximum for the number of available file descriptors.
  12. DAEMON_MAXFD = 1024
  13. # The standard I/O file descriptors are redirected to /dev/null by default.
  14. if (hasattr(os, "devnull")):
  15. REDIRECT_TO = os.devnull
  16. else:
  17. REDIRECT_TO = "/dev/null"
  18. class PIDFile(object):
  19. """Manages a pid file."""
  20. def __init__(self, pidfile):
  21. self.pidfile = pidfile
  22. def get_pid(self):
  23. """Get the process id stored in the pidfile."""
  24. pidfile_fh = file(self.pidfile, "r")
  25. pid = int(pidfile_fh.read().strip())
  26. pidfile_fh.close()
  27. return pid
  28. def check(self):
  29. """Check the status of the pidfile.
  30. If the pidfile exists, and the process is not running, it will
  31. remove the stale pidfile and continue as normal. If the process
  32. *is* running, it will exit the program with an error message.
  33. """
  34. if os.path.exists(self.pidfile) and os.path.isfile(self.pidfile):
  35. pid = self.get_pid()
  36. try:
  37. os.kill(pid, 0)
  38. except os.error, e:
  39. if e.errno == errno.ESRCH:
  40. sys.stderr.write("Stale pidfile exists. removing it.\n")
  41. self.remove()
  42. else:
  43. raise SystemExit("celeryd is already running.")
  44. def remove(self):
  45. """Remove the pidfile."""
  46. os.unlink(self.pidfile)
  47. def write(self, pid=None):
  48. """Write a pidfile.
  49. If ``pid`` is not specified the pid of the current process
  50. will be used.
  51. """
  52. if not pid:
  53. pid = os.getpid()
  54. pidfile_fh = file(self.pidfile, "w")
  55. pidfile_fh.write("%d\n" % pid)
  56. pidfile_fh.close()
  57. def remove_pidfile(pidfile):
  58. """Remove the pidfile."""
  59. os.unlink(pidfile)
  60. def daemonize(pidfile):
  61. """Detach a process from the controlling terminal and run it in the
  62. background as a daemon."""
  63. try:
  64. pid = os.fork()
  65. except OSError, e:
  66. raise Exception("%s [%d]" % (e.strerror, e.errno))
  67. if pid == 0: # child
  68. os.setsid()
  69. try:
  70. pid = os.fork() # second child
  71. except OSError, e:
  72. raise Exception("%s [%d]" % (e.strerror, e.errno))
  73. if pid == 0: # second child
  74. #os.chdir(DAEMON_WORKDIR)
  75. os.umask(DAEMON_UMASK)
  76. else: # parent (first child)
  77. pidfile.write(pid)
  78. os._exit(0)
  79. else: # root process
  80. os._exit(0)
  81. maxfd = resource.getrlimit(resource.RLIMIT_NOFILE)[1]
  82. if (maxfd == resource.RLIM_INFINITY):
  83. maxfd = DAEMON_MAXFD
  84. # Iterate through and close all file descriptors.
  85. for fd in range(0, maxfd):
  86. try:
  87. os.close(fd)
  88. except OSError:
  89. pass
  90. os.open(REDIRECT_TO, os.O_RDWR)
  91. # Duplicate standard input to standard output and standard error.
  92. os.dup2(0, 1)
  93. os.dup2(0, 2)
  94. return 0