Hey everyone,

I've seen references and questions in previous posts about how to spawn
a process to do some hefty processing. I had that need (processing of
1.6 million records at a time).

I created this code with help from other places on the net and it seems
to work pretty well. I'm putting it out there to get some further
testing and hope it is useful for someone else.

Usage is after the file.

Corey

daemonize.py:

#! /usr/bin/env python

import sys, os, time, tempfile
from signal import SIGTERM

def daemonize(stdout='/dev/null', stderr=None, stdin='/dev/null',
pidfile=None, startmsg = 'started with pid %s' ):
    '''
        This forks the current process into a daemon.
        The stdin, stdout, and stderr arguments are file names that
        will be opened and be used to replace the standard file
descriptors
        in sys.stdin, sys.stdout, and sys.stderr.
        These arguments are optional and default to /dev/null.
        Note that stderr is opened unbuffered, so
        if it shares a file with stdout then interleaved output
        may not appear in the order that you expect.
    '''
    # Do first fork.
    try:
        pid = os.fork()
        if pid > 0: sys.exit(0) # Exit first parent.
    except OSError, e:
        sys.stderr.write("fork #1 failed: (%d) %s\n" % (e.errno,
e.strerror))
        sys.exit(1)

    # Decouple from parent environment.
    os.chdir("/")
    os.umask(0)
    os.setsid()

    # Do second fork.
    try:
        pid = os.fork()
        if pid > 0: sys.exit(0) # Exit second parent.
    except OSError, e:
        sys.stderr.write("fork #2 failed: (%d) %s\n" % (e.errno,
e.strerror))
        sys.exit(1)

    # Open file descriptors and print start message
    if not stderr: stderr = stdout
    si = file(stdin, 'r')
    so = file(stdout, 'a+')
    se = file(stderr, 'a+', 0)

    pid = str(os.getpid())
    if startmsg:
        sys.stderr.write("\n%s\n" % startmsg % pid)
    sys.stderr.flush()
    if pidfile: file(pidfile,'w+').write("%s\n" % pid)

    # Redirect standard file descriptors.
    os.dup2(si.fileno(), sys.stdin.fileno())
    os.dup2(so.fileno(), sys.stdout.fileno())
    os.dup2(se.fileno(), sys.stderr.fileno())


def get_pid(pidfile):
    """
        Tries to get the current pid, given a pid file.

        Use this function to make sure a copy of this process isn't
already running.

        @param String pidfile   The path to the pid file
        @return Integer or None
    """
    try:
        pf  = file(pidfile,'r')
        pid = int(pf.read().strip())
        pf.close()
    except StandardError:
        pid = None
    if pid:
        mess = "Start aborted since pid file '%s' exists.\n"
        sys.stderr.write(mess % pidfile)
    return pid


if __name__ == "__main__":
    pid_file = '/tmp/%s.pid' % sys.argv[2]

    #########################
    # We don't want two of the same processes happening at the same
time for any reason
    #########################
    if get_pid(pid_file):
        f = open('/tmp/debuging.txt')
        f.write('Exiting because a pid file already exists.')
        f.close()
        sys.exit(1)

    #########################
    # Split ourselves from the parent process
    #########################
    daemonize(stdout='/tmp/%s-out.txt' % sys.argv[2],
stderr='/tmp/%s-err.txt' % sys.argv[2], startmsg='', pidfile=pid_file)
    print "daemonizing completed.\n"

    #########################
    # Generate the command to run
    #########################

    # We need to add the directory containing the project.
    # This command may be crude, but it gets the directory of the
directory of the script.
    thePath = os.path.dirname(os.path.dirname(sys.argv[0]))
    print "Adding %s to sys.path.\n" % thePath
    sys.path.append(thePath)

    print "Importing %s.\n" % sys.argv[1]
    m = __import__(sys.argv[1], globals(), locals(), '*')

    command = r'm.%s(' % sys.argv[2]
    for item in range(3,len(sys.argv)):
        arg = sys.argv[item].strip()

        # If we are passing tuples or lists, we don't want them quoted
        if arg[:1] == '(' or arg[:1] == '[':
            command += r' %s,' % arg
        else:
            command += r' "%s",' % arg

    command = command[:-1] + r')'

    ########################
    # Run the command
    ########################
    print "running %s" % command
    eval(command)

    ########################
    # Delete the pid file
    ########################
    os.remove(pid_file)
    sys.exit(0)

##### End daemonize.py

In your view you put in code similar to this:

# To import the file without creating a long delay, we are going to
split off the script
# ...daemonize it. The daemonize script will then call
importfile.start_import(theTempFile)
# The daemonizer.py script is in the main project folder...
daemonizer = "%s%s" % (settings.ROOT_DIR,'daemonize.py')
module = "pennysaver.distribution.importfile"
function = "start_import"
os.spawnvp(os.P_NOWAIT, daemonizer, [daemonizer, module, function,
theTempPath])

return render_to_response('import_progress.html')


--~--~---------~--~----~------------~-------~--~----~
You received this message because you are subscribed to the Google Groups 
"Django users" group.
To post to this group, send email to django-users@googlegroups.com
To unsubscribe from this group, send email to [EMAIL PROTECTED]
For more options, visit this group at 
http://groups.google.com/group/django-users
-~----------~----~----~----~------~----~------~--~---

Reply via email to