Home | History | Annotate | Download | only in pdist
      1 """RPC Server module."""
      2 
      3 import sys
      4 import socket
      5 import pickle
      6 from fnmatch import fnmatch
      7 from repr import repr
      8 
      9 
     10 # Default verbosity (0 = silent, 1 = print connections, 2 = print requests too)

     11 VERBOSE = 1
     12 
     13 
     14 class Server:
     15 
     16     """RPC Server class.  Derive a class to implement a particular service."""
     17 
     18     def __init__(self, address, verbose = VERBOSE):
     19         if type(address) == type(0):
     20             address = ('', address)
     21         self._address = address
     22         self._verbose = verbose
     23         self._socket = None
     24         self._socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
     25         self._socket.bind(address)
     26         self._socket.listen(1)
     27         self._listening = 1
     28 
     29     def _setverbose(self, verbose):
     30         self._verbose = verbose
     31 
     32     def __del__(self):
     33         self._close()
     34 
     35     def _close(self):
     36         self._listening = 0
     37         if self._socket:
     38             self._socket.close()
     39         self._socket = None
     40 
     41     def _serverloop(self):
     42         while self._listening:
     43             self._serve()
     44 
     45     def _serve(self):
     46         if self._verbose: print "Wait for connection ..."
     47         conn, address = self._socket.accept()
     48         if self._verbose: print "Accepted connection from %s" % repr(address)
     49         if not self._verify(conn, address):
     50             print "*** Connection from %s refused" % repr(address)
     51             conn.close()
     52             return
     53         rf = conn.makefile('r')
     54         wf = conn.makefile('w')
     55         ok = 1
     56         while ok:
     57             wf.flush()
     58             if self._verbose > 1: print "Wait for next request ..."
     59             ok = self._dorequest(rf, wf)
     60 
     61     _valid = ['192.16.201.*', '192.16.197.*', '132.151.1.*', '129.6.64.*']
     62 
     63     def _verify(self, conn, address):
     64         host, port = address
     65         for pat in self._valid:
     66             if fnmatch(host, pat): return 1
     67         return 0
     68 
     69     def _dorequest(self, rf, wf):
     70         rp = pickle.Unpickler(rf)
     71         try:
     72             request = rp.load()
     73         except EOFError:
     74             return 0
     75         if self._verbose > 1: print "Got request: %s" % repr(request)
     76         try:
     77             methodname, args, id = request
     78             if '.' in methodname:
     79                 reply = (None, self._special(methodname, args), id)
     80             elif methodname[0] == '_':
     81                 raise NameError, "illegal method name %s" % repr(methodname)
     82             else:
     83                 method = getattr(self, methodname)
     84                 reply = (None, apply(method, args), id)
     85         except:
     86             reply = (sys.exc_type, sys.exc_value, id)
     87         if id < 0 and reply[:2] == (None, None):
     88             if self._verbose > 1: print "Suppress reply"
     89             return 1
     90         if self._verbose > 1: print "Send reply: %s" % repr(reply)
     91         wp = pickle.Pickler(wf)
     92         wp.dump(reply)
     93         return 1
     94 
     95     def _special(self, methodname, args):
     96         if methodname == '.methods':
     97             if not hasattr(self, '_methods'):
     98                 self._methods = tuple(self._listmethods())
     99             return self._methods
    100         raise NameError, "unrecognized special method name %s" % repr(methodname)
    101 
    102     def _listmethods(self, cl=None):
    103         if not cl: cl = self.__class__
    104         names = cl.__dict__.keys()
    105         names = filter(lambda x: x[0] != '_', names)
    106         names.sort()
    107         for base in cl.__bases__:
    108             basenames = self._listmethods(base)
    109             basenames = filter(lambda x, names=names: x not in names, basenames)
    110             names[len(names):] = basenames
    111         return names
    112 
    113 
    114 from security import Security
    115 
    116 
    117 class SecureServer(Server, Security):
    118 
    119     def __init__(self, *args):
    120         apply(Server.__init__, (self,) + args)
    121         Security.__init__(self)
    122 
    123     def _verify(self, conn, address):
    124         import string
    125         challenge = self._generate_challenge()
    126         conn.send("%d\n" % challenge)
    127         response = ""
    128         while "\n" not in response and len(response) < 100:
    129             data = conn.recv(100)
    130             if not data:
    131                 break
    132             response = response + data
    133         try:
    134             response = string.atol(string.strip(response))
    135         except string.atol_error:
    136             if self._verbose > 0:
    137                 print "Invalid response syntax", repr(response)
    138             return 0
    139         if not self._compare_challenge_response(challenge, response):
    140             if self._verbose > 0:
    141                 print "Invalid response value", repr(response)
    142             return 0
    143         if self._verbose > 1:
    144             print "Response matches challenge.  Go ahead!"
    145         return 1
    146