[Zope-Checkins] SVN: zdaemon/trunk/ Added reference documentation and some edge-case tests.

Jim Fulton jim at zope.com
Mon Jan 8 06:59:15 EST 2007


Log message for revision 71789:
  Added reference documentation and some edge-case tests.
  
  Refined code in the control (user-interface) progream to detect
  sooner if the deamon manager has exited.
  
  Updated release info.
  

Changed:
  U   zdaemon/trunk/CHANGES.txt
  U   zdaemon/trunk/README.txt
  U   zdaemon/trunk/setup.py
  U   zdaemon/trunk/src/zdaemon/README.txt
  U   zdaemon/trunk/src/zdaemon/tests/tests.py
  U   zdaemon/trunk/src/zdaemon/zdctl.py

-=-
Modified: zdaemon/trunk/CHANGES.txt
===================================================================
--- zdaemon/trunk/CHANGES.txt	2007-01-08 00:10:56 UTC (rev 71788)
+++ zdaemon/trunk/CHANGES.txt	2007-01-08 11:59:13 UTC (rev 71789)
@@ -4,12 +4,6 @@
 To-Dos
 ======
 
-Tests:
-
-- non-daemon mode
-
-- no infinite loop when program fails on start
-
 More docs:
 
 - Document/demonstrate some important features, such as:
@@ -18,9 +12,6 @@
 
   - working directory
 
-- Reference docs
-
-
 Features
 
 - environment variables

Modified: zdaemon/trunk/README.txt
===================================================================
--- zdaemon/trunk/README.txt	2007-01-08 00:10:56 UTC (rev 71788)
+++ zdaemon/trunk/README.txt	2007-01-08 11:59:13 UTC (rev 71789)
@@ -1,8 +1,9 @@
-``zdaemon`` process controller
-==============================
+****************************************************
+``zdaemon`` process controller for Unix-ased systems
+****************************************************
 
 Overview
---------
+********
 
 'zdaemon' is a Python package which provides APIs for managing spplications
 run as daemons.  Its principal use to date has been to manage the application

Modified: zdaemon/trunk/setup.py
===================================================================
--- zdaemon/trunk/setup.py	2007-01-08 00:10:56 UTC (rev 71788)
+++ zdaemon/trunk/setup.py	2007-01-08 11:59:13 UTC (rev 71789)
@@ -16,8 +16,7 @@
 
 entry_points = """
 [console_scripts]
-zdctl = zdaemon.zdctl:main
-zdrun = zdaemon.zdrun:main
+zdaemon = zdaemon.zdctl:main
 """
 
 def read(*rnames):
@@ -29,6 +28,8 @@
         zip_safe=False,
         entry_points=entry_points,
         include_package_data = True,
+        install_requires=["ZConfig"],
+        tests_require=["zope.testing"],
         )
 except ImportError:
     from distutils.core import setup
@@ -37,10 +38,11 @@
 name = "zdaemon"
 setup(
     name=name,
-    version="1.4a2",
+    version="2.0a1",
     url="http://www.python.org/pypi/zdaemon",
     license="ZPL 2.1",
-    description="Daemon process control library and tools",
+    description=
+    "Daemon process control library and tools for Unix-bases systems",
     author="Zope Corporation and Contributors",
     author_email="zope3-dev at zope.org",
     long_description=(
@@ -48,13 +50,16 @@
         + '\n' +
         read('CHANGES.txt')
         + '\n' +
+        'Detailed Documentation\n'
+        '**********************\n'
+        + '\n' +
+        read('src', 'zdaemon', 'README.txt')
+        + '\n' +
         'Download\n'
         '**********************\n'
         ),
 
     packages=["zdaemon", "zdaemon.tests"],
     package_dir={"": "src"},
-    install_requires=["ZConfig"],
-    tests_require=["zope.testing"],
     
     **setuptools_options)

Modified: zdaemon/trunk/src/zdaemon/README.txt
===================================================================
--- zdaemon/trunk/src/zdaemon/README.txt	2007-01-08 00:10:56 UTC (rev 71788)
+++ zdaemon/trunk/src/zdaemon/README.txt	2007-01-08 11:59:13 UTC (rev 71789)
@@ -137,3 +137,192 @@
     >>> system("./zdaemon -Cconf stop")
     daemon process stopped
 
+Reference Documentation
+-----------------------
+
+The following options are available for use in the runner section of
+configuration files and as command-line options.
+
+program
+        Command-line option: -p or --program
+
+        This option gives the command used to start the subprocess
+        managed by zdaemon.  This is currently a simple list of
+        whitespace-delimited words. The first word is the program
+        file, subsequent words are its command line arguments.  If the
+        program file contains no slashes, it is searched using $PATH.
+        (Note that there is no way to to include whitespace in the program
+        file or an argument, and under certain circumstances other
+        shell metacharacters are also a problem.)
+
+socket-name
+        Command-line option: -s or --socket-name.
+
+        The pathname of the Unix domain socket used for communication
+        between the zdaemon command-line tool and a deamon-management
+        process.  The default is relative to the current directory in
+        which zdaemon is started.  You want to specify
+        an absolute pathname here.
+
+        This defaults to "zdsock", which is created in the directory
+        in which zdrun is started.
+
+daemon
+        Command-line option: -d or --daemon.
+
+        If this option is true, zdaemon runs in the background as a
+        true daemon.  It forks a child process which becomes the
+        subprocess manager, while the parent exits (making the shell
+        that started it believe it is done).  The child process also
+        does the following:
+
+        - if the directory option is set, change into that directory
+
+        - redirect stdin, stdout and stderr to /dev/null
+
+        - call setsid() so it becomes a session leader
+
+        - call umask() with specified value
+
+        The default for this option is on by default.  The
+        command-line option therefore has no effect.  To disable
+        daemon mode, you must use a configuration file::
+
+          <runner>
+            program sleep 1
+            daemon off
+          </runner>
+
+directory
+        Command-line option: -z or --directory.
+
+        If the daemon option is true (default), this option can
+        specify a directory into which zdrun.py changes as part of the
+        "daemonizing".  If the daemon option is false, this option is
+        ignored.
+
+backoff-limit
+        Command-line option: -b or --backoff-limit.
+
+        When the subprocess crashes, zdaemon inserts a one-second
+        delay before it restarts it.  When the subprocess crashes
+        again right away, the delay is incremented by one second, and
+        so on.  What happens when the delay has reached the value of
+        backoff-limit (in seconds), depends on the value of the
+        forever option.  If forever is false, zdaemon gives up at
+        this point, and exits.  An always-crashing subprocess will
+        have been restarted exactly backoff-limit times in this case.
+        If forever is true, zdaemon continues to attempt to restart
+        the process, keeping the delay at backoff-limit seconds.
+
+        If the subprocess stays up for more than backoff-limit
+        seconds, the delay is reset to 1 second.
+
+        This defaults to 10.
+
+forever
+        Command-line option: -f or --forever.
+
+        If this option is true, zdaemon will keep restarting a
+        crashing subprocess forever.  If it is false, it will give up
+        after backoff-limit crashes in a row.  See the description of
+        backoff-limit for details.
+
+        This is disabled by default.
+
+exit-codes
+        Command-line option: -x or --exit-codes.
+
+        This defaults to 0,2.
+
+        If the subprocess exits with an exit status that is equal to
+        one of the integers in this list, zdaemon will not restart
+        it.  The default list requires some explanation.  Exit status
+        0 is considered a willful successful exit; the ZEO and Zope
+        server processes use this exit status when they want to stop
+        without being restarted.  (Including in response to a
+        SIGTERM.)  Exit status 2 is typically issued for command line
+        syntax errors; in this case, restarting the program will not
+        help!
+
+        NOTE: this mechanism overrides the backoff-limit and forever
+        options; i.e. even if forever is true, a subprocess exit
+        status code in this list makes zdaemon give up.  To disable
+        this, change the value to an empty list.
+
+user
+        Command-line option: -u or --user.
+
+        When zdaemon is started by root, this option specifies the
+        user as who the the zdaemon process (and hence the daemon
+        subprocess) will run.  This can be a user name or a numeric
+        user id.  Both the user and the group are set from the
+        corresponding password entry, using setuid() and setgid().
+        This is done before zdaemon does anything else besides
+        parsing its command line arguments.
+
+        NOTE: when zdaemon is not started by root, specifying this
+        option is an error.  (XXX This may be a mistake.)
+
+        XXX The zdaemon event log file may be opened *before*
+        setuid() is called.  Is this good or bad?
+
+umask
+        Command-line option: -m or --umask.
+
+        When daemon mode is used, this option specifies the octal umask
+        of the subprocess.
+
+default-to-interactive
+        If this option is true, zdaemon enters interactive mode
+        when it is invoked without a positional command argument.  If
+        it is false, you must use the -i or --interactive command line
+        option to zdaemon to enter interactive mode.
+
+        This is enabled by default.
+
+logfile
+        This option specifies a log file that is the default target of
+        the "logtail" zdaemon command.
+
+        NOTE: This is NOT the log file to which zdaemon writes its
+        logging messages!  That log file is specified by the
+        <eventlog> section described below.
+
+transcript
+        The name of a file in which a transcript of all output from
+        the command being run will be written to when daemonized.
+
+        If not specified, output from the command will be discarded.
+
+        This only takes effect when the "daemon" option is enabled.
+
+prompt
+         The prompt shown by the controller program.  The default must
+         be provided by the application.
+
+(Note that a few other options are available to support old
+configuration files, but aren't needed any more and can generally be
+ignored.)
+
+In addition to the runner section, you can use an eventlog section
+that specified one or more logfile subsections::
+
+    <eventlog>
+      <logfile>
+        path /var/log/foo/foo.log
+      </logfile>
+
+      <logfile>
+        path STDOUT
+      </logfile>
+    </eventlog>
+
+In this example, log output is sent to a file and to standard out.
+Log output from zdaemon usually isn't very interesting but can be
+handy for debugging.
+
+
+
+
+

Modified: zdaemon/trunk/src/zdaemon/tests/tests.py
===================================================================
--- zdaemon/trunk/src/zdaemon/tests/tests.py	2007-01-08 00:10:56 UTC (rev 71788)
+++ zdaemon/trunk/src/zdaemon/tests/tests.py	2007-01-08 11:59:13 UTC (rev 71789)
@@ -31,6 +31,44 @@
     zconfig_loc = pkg_resources.working_set.find(
         pkg_resources.Requirement.parse('ZConfig')).location
 
+
+def make_sure_non_daemon_mode_doesnt_hang_when_program_exits():
+    """
+    The whole awhile bit that waits for a program to start
+    whouldn't be used on non-daemopn mode.
+
+    >>> open('conf', 'w').write(
+    ... '''
+    ... <runner>
+    ...   program sleep 1
+    ...   daemon off
+    ... </runner>
+    ... ''')
+
+    >>> system("./zdaemon -Cconf start")
+
+    """
+
+def dont_hang_when_program_doesnt_start():
+    """
+    If a program doesn't start, we don't want to wait for ever.
+
+    >>> open('conf', 'w').write(
+    ... '''
+    ... <runner>
+    ...   program sleep
+    ...   backoff-limit 2
+    ... </runner>
+    ... ''')
+
+    >>> system("./zdaemon -Cconf start")
+    . . 
+    Daemon manager not running.
+
+    """
+    
+
+
 def setUp(test):
     test.globs['_td'] = td = []
     here = os.getcwd()
@@ -60,8 +98,15 @@
     print o.read(),
 
 
+
 def test_suite():
     return unittest.TestSuite((
+        doctest.DocTestSuite(
+            setUp=setUp, tearDown=tearDown,
+            checker=renormalizing.RENormalizing([
+                (re.compile('pid=\d+'), 'pid=NNN'),
+                ])
+        ),
         doctest.DocFileSuite(
             '../README.txt',
             setUp=setUp, tearDown=tearDown,

Modified: zdaemon/trunk/src/zdaemon/zdctl.py
===================================================================
--- zdaemon/trunk/src/zdaemon/zdctl.py	2007-01-08 00:10:56 UTC (rev 71788)
+++ zdaemon/trunk/src/zdaemon/zdctl.py	2007-01-08 11:59:13 UTC (rev 71789)
@@ -177,16 +177,22 @@
 
     def awhile(self, cond, msg):
         n = 0
+        was_running = True
         try:
-            self.get_status()
+            if self.get_status():
+                was_running = True
+                
             while not cond():
                 sys.stdout.write(". ")
                 sys.stdout.flush()
                 time.sleep(1)
                 n += 1
-                if not self.get_status() and n > 10:
+                if self.get_status():
+                    was_running = True
+                elif was_running or n > 10:
                     print "\nDaemon manager not running."
                     return
+
         except KeyboardInterrupt:
             print "^C"
         else:



More information about the Zope-Checkins mailing list