Source code for pyrocore.util.osmagic

# -*- coding: utf-8 -*-
# pylint: disable=I0011
""" Platform Specific Incantations.

    Copyright (c) 2011 The PyroScope Project <pyroscope.project@gmail.com>
"""
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with this program; if not, write to the Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
from __future__ import absolute_import

import sys
import time
import errno
import signal
import logging

from pyrocore.util import os

def _write_pidfile(pidfile):
    """ Write file with current process ID.
    """
    pid = str(os.getpid())
    handle = open(pidfile, 'w')
    try:
        handle.write("%s\n" % pid)
    finally:
        handle.close()


[docs]def check_process(pidfile): """ Read pid file and check process status. Return (running, pid). """ # Check pid file try: handle = open(pidfile, 'r') except IOError as exc: if exc.errno == errno.ENOENT: # pid file disappeared return False, 0 raise try: pid = int(handle.read().strip(), 10) except (TypeError, ValueError) as exc: raise EnvironmentError("Invalid PID file '%s' (%s), won't start!" % (pidfile, exc)) finally: handle.close() # Check process try: os.kill(pid, 0) except EnvironmentError as exc: return False, pid else: return True, pid
[docs]def guard(pidfile, guardfile=None): """ Raise an EnvironmentError when the "guardfile" doesn't exist, or the process with the ID found in "pidfile" is still active. """ # Check guard if guardfile and not os.path.exists(guardfile): raise EnvironmentError("Guard file '%s' not found, won't start!" % guardfile) if os.path.exists(pidfile): running, pid = check_process(pidfile) if running: raise EnvironmentError("Daemon process #%d still running, won't start!" % pid) else: logging.getLogger("daemonize").info("Process #%d disappeared, continuing..." % pid) # Keep race condition window small, by immediately writing launcher process ID _write_pidfile(pidfile)
[docs]def daemonize(pidfile=None, logfile=None, sync=True): """ Fork the process into the background. @param pidfile: Optional PID file path. @param sync: Wait for parent process to disappear? @param logfile: Optional name of stdin/stderr log file or stream. """ log = logging.getLogger("daemonize") ppid = os.getpid() try: pid = os.fork() if pid > 0: log.debug("Parent exiting (PID %d, CHILD %d)" % (ppid, pid)) sys.exit(0) except OSError as exc: log.critical("fork #1 failed (PID %d): (%d) %s\n" % (os.getpid(), exc.errno, exc.strerror)) sys.exit(1) ##os.chdir("/") ##os.umask(0022) os.setsid() try: pid = os.fork() if pid > 0: log.debug("Session leader exiting (PID %d, PPID %d, DEMON %d)" % (os.getpid(), ppid, pid)) sys.exit(0) except OSError as exc: log.critical("fork #2 failed (PID %d): (%d) %s\n" % (os.getpid(), exc.errno, exc.strerror)) sys.exit(1) if pidfile: _write_pidfile(pidfile) def sig_term(*dummy): "Handler for SIGTERM." sys.exit(0) stdin = open("/dev/null", "r") os.dup2(stdin.fileno(), sys.stdin.fileno()) signal.signal(signal.SIGTERM, sig_term) if logfile: try: logfile + "" except TypeError: if logfile.fileno() != sys.stdout.fileno(): os.dup2(logfile.fileno(), sys.stdout.fileno()) if logfile.fileno() != sys.stderr.fileno(): os.dup2(logfile.fileno(), sys.stderr.fileno()) else: log.debug("Redirecting stdout / stderr to %r" % logfile) loghandle = open(logfile, "a+") os.dup2(loghandle.fileno(), sys.stdout.fileno()) os.dup2(loghandle.fileno(), sys.stderr.fileno()) loghandle.close() if sync: # Wait for 5 seconds at most, in 10ms steps polling = 5, .01 for _ in range(int(polling[0] * 1 / polling[1])): try: os.kill(ppid, 0) except OSError: break else: time.sleep(polling[1]) log.debug("Process detached (PID %d)" % os.getpid())