[Zope-Checkins] CVS: ZODB3/zdaemon - zdaemon.py:1.7

Guido van Rossum guido@python.org
Mon, 11 Nov 2002 15:17:45 -0500


Update of /cvs-repository/ZODB3/zdaemon
In directory cvs.zope.org:/tmp/cvs-serv15163

Modified Files:
	zdaemon.py 
Log Message:
Refactoring of command handling -- commands are now defined by methods
whose name begins with 'cmd_'.  The signature is always
cmd_foo(self, args) where args is a list of arguments and args[0] is
'foo'.


=== ZODB3/zdaemon/zdaemon.py 1.6 => 1.7 ===
--- ZODB3/zdaemon/zdaemon.py:1.6	Mon Nov 11 12:38:39 2002
+++ ZODB3/zdaemon/zdaemon.py	Mon Nov 11 15:17:45 2002
@@ -20,9 +20,10 @@
   [command] -- the command to send to the daemon manager (default "status")
 
 Client commands are:
-  status -- report application status
-  kill [signal] -- send the given signal number to the application
-                   (default signal is 14, i.e. SIGTERM)
+  help -- return command help
+  status -- report application status (this is the default command)
+  kill [signal] -- send a signal to the application
+                   (default signal is SIGTERM)
 
 This daemon manager has two purposes: it restarts the application when
 it dies, and (when requested to do so with the -d option) it runs the
@@ -51,18 +52,14 @@
 """
 XXX TO DO
 
-- Refactor the client/server code to be less ugly; avoid blocking
-  recv() calls; don't assume send() always sends everything.
+- Refactor the readcommand() to avoid blocking recv() calls.
 
 - Rethink client commands; maybe start/restart/stop make more sense?
   (Still need a way to send an arbitrary signal)
 
-- Should it be possible to change other parameters from the client?
-  (e.g. filename, command, backoff etc.)
-
 - Do the governor without actual sleeps, using event scheduling etc.
 
-- Add docstrings
+- Add docstrings.
 
 """
 
@@ -287,62 +284,98 @@
                     self.appid = 0
                 self.reportstatus(wpid, wsts)
             if r:
-                try:
-                    self.readcommand()
-                except:
-                    self.problem("Exception in readcommand()")
+                self.readcommand()
+
+    conn = None
 
     def readcommand(self):
-        conn, addr = self.controlsocket.accept()
         try:
-            data = conn.recv(1000)
-            if not data:
-                conn.send("No input\n")
-                return
-            line = data
-            while "\n" not in line:
+            conn, addr = self.controlsocket.accept()
+            self.conn = conn
+            try:
                 data = conn.recv(1000)
                 if not data:
-                    conn.send("Incomplete input\n")
+                    self.sendreply("No input")
                     return
-                line += data
-            lines = line.split("\n")
-            words = lines[0].split()
-            if not words:
-                conn.send("Empty command\n")
-                return
-            command = words[0]
-            if command == "status":
-                if not self.appid:
-                    status = "not running"
-                else:
-                    status = "running"
-                conn.send("Application %s\n" % status +
-                          "pid=%d\n" % self.appid +
-                          "filename=%s\n" % repr(self.filename) +
-                          "args=%s\n" % repr(self.args))
-            elif command == "kill":
-                if words[1:]:
-                    try:
-                        sig = int(words[1])
-                    except:
-                        conn.send("Bad signal %s\n" % repr(words[1]))
+                line = data
+                while "\n" not in line:
+                    data = conn.recv(1000)
+                    if not data:
+                        self.sendreply("Input not terminated by newline")
                         return
-                else:
-                    sig = signal.SIGTERM
-                if not self.appid:
-                    conn.send("Application not running\n")
-                else:
-                    try:
-                        os.kill(self.appid, sig)
-                    except os.error, msg:
-                        conn.send("Kill %d failed: %s\n" % (sig, str(msg)))
-                    else:
-                        conn.send("Signal %d sent\n" % sig)
+                    line += data
+                self.docommand(line)
+            finally:
+                conn.close()
+            self.conn = None
+        except socket.error, msg:
+            self.problem("socket error: %s" % str(msg),
+                         error=sys.exc_info())
+
+    def docommand(self, line):
+        lines = line.split("\n")
+        args = lines[0].split()
+        if not args:
+            self.sendreply("Empty command")
+            return
+        command = args[0]
+        methodname = "cmd_" + command
+        method = getattr(self, methodname, None)
+        if method:
+            method(args)
+        else:
+            self.sendreply("Unknown command %s" % (`args[0]`))
+
+    def cmd_kill(self, args):
+        if args[1:]:
+            try:
+                sig = int(args[1])
+            except:
+                self.sendreply("Bad signal %s" % repr(args[1]))
+                return
+        else:
+            sig = signal.SIGTERM
+        if not self.appid:
+            self.sendreply("Application not running")
+        else:
+            try:
+                os.kill(self.appid, sig)
+            except os.error, msg:
+                self.sendreply("Kill %d failed: %s" % (sig, str(msg)))
             else:
-                conn.send("Unknown command %s\n" % (`words[0]`))
-        finally:
-            conn.close()
+                self.sendreply("Signal %d sent" % sig)
+
+    def cmd_status(self, args):
+        if not self.appid:
+            status = "stopped"
+        else:
+            status = "running"
+        self.sendreply("status=%s\n" % status +
+                       "manager=%d\n" % os.getpid() + 
+                       "application=%d\n" % self.appid +
+                       "filename=%s\n" % repr(self.filename) +
+                       "args=%s\n" % repr(self.args))
+
+    def cmd_help(self, args):
+        self.sendreply(
+            "Available commands:\n"
+            "  help -- return command help\n"
+            "  status -- report application status (default command)\n"
+            "  kill [signal] -- send a signal to the application\n"
+            "                   (default signal is SIGTERM)\n"
+            )
+
+    def sendreply(self, msg):
+        if not msg.endswith("\n"):
+            msg = msg + "\n"
+        conn = self.conn
+        if hasattr(conn, "sendall"):
+            conn.sendall(msg)
+        else:
+            # This is quadratic, but msg is rarely more than 100 bytes :-)
+            while msg:
+                sent = conn.send(msg)
+                msg = msg[sent:]
 
     backoff = 0
     lasttime = None
@@ -402,10 +435,8 @@
             else:
                 self.warning(msg)
         elif os.WIFSIGNALED(sts):
-            signum = os.WTERMSIG(sts)
-            signame = self.signame(signum)
-            msg = ("pid %d: terminated by signal %s(%s)" %
-                   (pid, signame, signum))
+            sig = os.WTERMSIG(sts)
+            msg = ("pid %d: terminated by %s" % (pid, self.signame(sig)))
             if hasattr(os, "WCOREDUMP"):
                 iscore = os.WCOREDUMP(sts)
             else:
@@ -428,7 +459,7 @@
 
         if self.signames is None:
             self.setupsignames()
-        return self.signames.get(sig, "unknown")
+        return self.signames.get(sig) or "signal %d" % sig
 
     def setupsignames(self):
             self.signames = {}
@@ -470,17 +501,17 @@
     def warning(self, msg):
         self.log(msg, zLOG.WARNING)
 
-    def problem(self, msg):
-        self.log(msg, zLOG.ERROR)
+    def problem(self, msg, error=None):
+        self.log(msg, zLOG.ERROR, error)
 
-    def panic(self, msg):
-        self.log(msg, zLOG.PANIC)
+    def panic(self, msg, error=None):
+        self.log(msg, zLOG.PANIC, error)
 
     def getsubsystem(self):
         return "ZD:%d" % os.getpid()
 
-    def log(self, msg, severity=zLOG.INFO):
-        zLOG.LOG(self.getsubsystem(), severity, msg)
+    def log(self, msg, severity=zLOG.INFO, error=None):
+        zLOG.LOG(self.getsubsystem(), severity, msg, "", error)
 
 def main(args=None):
     d = Daemonizer()