Clean up Python server implementation, guided by pylint
authordbs <dbs@9efc2488-bf62-4759-914b-345cdb29e865>
Tue, 11 Jan 2011 22:59:18 +0000 (22:59 +0000)
committerdbs <dbs@9efc2488-bf62-4759-914b-345cdb29e865>
Tue, 11 Jan 2011 22:59:18 +0000 (22:59 +0000)
Add some docstrings, keep line lengths at 80 or less (MORE
WHITESPACE!), remove some dead code, and add a hint to check
max_children config value if we're bumping up against the
limit.

git-svn-id: svn://svn.open-ils.org/OpenSRF/trunk@2141 9efc2488-bf62-4759-914b-345cdb29e865

src/python/osrf/server.py

index 0255b91..8081a39 100644 (file)
@@ -1,3 +1,6 @@
+"""
+Implements an OpenSRF forking request server
+"""
 # -----------------------------------------------------------------------
 # Copyright (C) 2008-2010  Equinox Software, Inc.
 # Bill Erickson <erickson@esilibrary.com>
@@ -18,8 +21,9 @@
 # 02110-1301, USA
 # -----------------------------------------------------------------------
 
-import os, sys, threading, fcntl, socket, errno, signal, time
-import osrf.log, osrf.conf, osrf.net, osrf.system, osrf.stack, osrf.app, osrf.const
+import os, sys, fcntl, socket, errno, signal, time
+import osrf.log, osrf.conf, osrf.net, osrf.system
+import osrf.stack, osrf.app, osrf.const
 
 
 # used to define the size of the PID/size leader in 
@@ -32,6 +36,7 @@ class Controller(object):
     '''
 
     def __init__(self, service):
+        '''Initialize the Controller object'''
         self.service = service # service name
         self.max_requests = 0 # max child requests
         self.max_children = 0 # max num of child processes
@@ -49,10 +54,6 @@ class Controller(object):
         self.read_status, self.write_status = socket.socketpair()
         self.read_status.setblocking(0)
 
-    def load_app(self):
-        settings = osrf.set.get('activeapps.%s' % self.service)
-        
-
     def cleanup(self):
         ''' Closes management sockets, kills children, reaps children, exits '''
 
@@ -77,26 +78,32 @@ class Controller(object):
 
     def handle_signals(self):
         ''' Installs SIGINT and SIGTERM handlers '''
+
         def handler(signum, frame):
+            ''' Handler implementation '''
             self.cleanup()
+
         signal.signal(signal.SIGINT, handler)
         signal.signal(signal.SIGTERM, handler)
 
 
     def run(self):
+        ''' Run the OpenSRF service, spawning and reaping children '''
 
         osrf.net.get_network_handle().disconnect()
         osrf.net.clear_network_handle()
         self.spawn_children()
         self.handle_signals()
 
-        time.sleep(.5) # give children a chance to connect before we start taking data
+        # give children a chance to connect before we start taking data
+        time.sleep(.5)
         self.osrf_handle = osrf.system.System.net_connect(
             resource = '%s_listener' % self.service,
             service = self.service
         )
 
-        # clear the recv callback so inbound messages do not filter through the opensrf stack
+        # clear the recv callback so inbound messages do not filter 
+        # through the opensrf stack
         self.osrf_handle.receive_callback = None
 
         # connect to our listening routers
@@ -115,14 +122,21 @@ class Controller(object):
                 if len(self.idle_list) > 0:
                     child = self.idle_list.pop() 
                     self.active_list.append(child)
-                    osrf.log.log_internal("server: sending data to available child %d" % child.pid)
+                    osrf.log.log_internal(
+                        "server: sending data to available child %d" % child.pid
+                    )
 
                 elif self.num_children < self.max_children:
                     child = self.spawn_child(True)
-                    osrf.log.log_internal("server: sending data to new child %d" % child.pid)
+                    osrf.log.log_internal(
+                        "server: sending data to new child %d" % child.pid
+                    )
 
                 else:
-                    osrf.log.log_warn("server: no children available, waiting...")
+                    osrf.log.log_warn("server: no children available, \
+waiting... consider increasing max_children for this application higher than \
+%d in the  OpenSRF configuration if this message occurs frequently" \
+                        % self.max_children)
                     child = self.check_status(True)
 
                 self.write_child(child, data)
@@ -130,8 +144,10 @@ class Controller(object):
         except KeyboardInterrupt:
             osrf.log.log_info("server: exiting with keyboard interrupt")
 
-        except Exception, e: 
-            osrf.log.log_error("server: exiting with exception: %s" % e.message)
+        except Exception, exc: 
+            osrf.log.log_error(
+                "server: exiting with exception: %s" % exc.message
+            )
 
         finally:
             self.cleanup()
@@ -143,8 +159,11 @@ class Controller(object):
         try:
             child.write_data.sendall(data)
 
-        except Exception, e:
-            osrf.log.log_error("server: error sending data to child %d: %s" % (child.pid, str(e)))
+        except Exception, ex:
+            osrf.log.log_error(
+                "server: error sending data to child %d: %s" 
+                % (child.pid, str(ex))
+            )
             self.cleanup_child(child.pid, True)
             return False
 
@@ -167,11 +186,13 @@ class Controller(object):
             try:
                 pid = self.read_status.recv(SIZE_PAD)
 
-            except socket.error, e:
-                if e.args[0] == errno.EAGAIN:
+            except socket.error, exc:
+                if exc.args[0] == errno.EAGAIN:
                     break # no data left to read in nonblocking mode
 
-                osrf.log.log_error("server: child status check failed: %s" % str(e))
+                osrf.log.log_error(
+                    "server: child status check failed: %s" % str(exc)
+                )
                 if not wait or ret_child:
                     break
 
@@ -183,9 +204,12 @@ class Controller(object):
 
             if pid:
                 child = self.pid_map[int(pid)]
-                osrf.log.log_internal("server: child process %d reporting for duty" % child.pid)
+                osrf.log.log_internal(
+                    "server: child process %d reporting for duty" % child.pid
+                )
                 if wait and ret_child is None:
-                    # caller is waiting for a free child, leave it in the active list
+                    # caller is waiting for a free child;
+                    # leave it in the active list
                     ret_child = child
                 else:
                     self.active_list.remove(child)
@@ -195,7 +219,9 @@ class Controller(object):
         
 
     def reap_children(self, done=False):
-        ''' Uses waitpid() to reap the children.  If necessary, new children are spawned '''
+        '''
+        Uses waitpid() to reap the children. If necessary, spawns new children.
+        '''
 
         options = 0
         if not done: 
@@ -217,6 +243,11 @@ class Controller(object):
                 return
 
     def cleanup_child(self, pid, kill=False):
+        '''
+        Removes the child from the active or idle list.
+
+        Kills the process if requested.
+        '''
 
         if kill:
             os.kill(pid, signal.SIGKILL)
@@ -248,15 +279,25 @@ class Controller(object):
         child = Child(self)
         child.read_data, child.write_data = socket.socketpair()
         child.pid = os.fork()
+        sys.stdin.close()
+        sys.stdin = open(os.devnull, 'r')
+        sys.stdout.close()
+        sys.stdout = open(os.devnull, 'w')
+        sys.stderr.close()
+        sys.stderr = open(os.devnull, 'w')
+
 
-        if child.pid: # parent process
+        if child.pid:
             self.num_children += 1
             self.pid_map[child.pid] = child
             if active:
                 self.active_list.append(child)
             else:
                 self.idle_list.append(child)
-            osrf.log.log_internal("server: %s spawned child %d : %d total" % (self.service, child.pid, self.num_children))
+            osrf.log.log_internal(
+                "server: %s spawned child %d : %d total" 
+                % (self.service, child.pid, self.num_children)
+            )
             return child
         else:
             child.pid = os.getpid()
@@ -318,11 +359,22 @@ class Child(object):
     ''' Models a single child process '''
 
     def __init__(self, controller):
-        self.controller = controller # our Controller object
-        self.num_requests = 0 # how many requests we've served so far
-        self.read_data = None # the child reads data from the controller on this socket
-        self.write_data = None # the controller sends data to the child on this socket 
-        self.pid = 0 # my process id
+        ''' Initializes child process instance '''
+
+        # our Controller object
+        self.controller = controller
+
+        # how many requests we've served so far
+        self.num_requests = 0
+
+        # the child reads data from the controller on this socket
+        self.read_data = None
+
+        # the controller sends data to the child on this socket 
+        self.write_data = None
+
+        # my process id
+        self.pid = 0
 
     def run(self):
         ''' Loops, processing data, until max_requests is reached '''
@@ -339,10 +391,12 @@ class Child(object):
                     buf = None
                     try:
                         buf = self.read_data.recv(2048)
-                    except socket.error, e:
-                        if e.args[0] == errno.EAGAIN:
+                    except socket.error, exc:
+                        if exc.args[0] == errno.EAGAIN:
                             break
-                        osrf.log.log_error("server: child data read failed: %s" % str(e))
+                        osrf.log.log_error(
+                            "server: child data read failed: %s" % str(exc)
+                        )
                         osrf.app.Application.application.child_exit()
                         return
 
@@ -355,7 +409,9 @@ class Child(object):
                 osrf.log.log_internal("server: child received message: " + data)
 
                 osrf.net.get_network_handle().flush_inbound_data()
-                session = osrf.stack.push(osrf.net.NetworkMessage.from_xml(data))
+                session = osrf.stack.push(
+                    osrf.net.NetworkMessage.from_xml(data)
+                )
                 self.keepalive_loop(session)
 
                 self.num_requests += 1
@@ -375,6 +431,11 @@ class Child(object):
         osrf.app.Application.application.child_exit()
 
     def keepalive_loop(self, session):
+        '''
+        Keeps session alive while client is connected.
+
+        If timeout occurs, session disconnects and gets cleaned up.
+        '''
         keepalive = self.controller.keepalive
 
         while session.state == osrf.const.OSRF_APP_SESSION_CONNECTED:
@@ -382,14 +443,19 @@ class Child(object):
             status = session.wait(keepalive)
 
             if session.state == osrf.const.OSRF_APP_SESSION_DISCONNECTED:
-                osrf.log.log_internal("server: client sent disconnect, exiting keepalive")
+                osrf.log.log_internal(
+                    "server: client sent disconnect, exiting keepalive"
+                )
                 break
 
-            if status is None: # no msg received before keepalive timeout expired
+            # if no msg received before keepalive timeout expired
+            if status is None:
 
                 osrf.log.log_info(
-                    "server: no request was received in %d seconds from %s, exiting stateful session" % (
-                    session.remote_id, int(keepalive)));
+                    "server: no request was received in %d seconds from %s, "
+                    "exiting stateful session"
+                    % (session.remote_id, int(keepalive))
+                )
 
                 session.send_status(
                     session.thread,