+"""provides xmlrpc server functionality for wxPython applications via a mixin class
+
+**Some Notes:**
+
+ 1) The xmlrpc server runs in a separate thread from the main GUI
+ application, communication between the two threads using a custom
+ event (see the Threads demo in the wxPython docs for more info).
+
+ 2) Neither the server nor the client are particularly smart about
+ checking method names. So it's easy to shoot yourself in the foot
+ by calling improper methods. It would be pretty easy to add
+ either a list of allowed methods or a list of forbidden methods.
+
+ 3) Authentication of xmlrpc clients is *not* performed. I think it
+ would be pretty easy to do this in a hacky way, but I haven't done
+ it yet.
+
+ 4) See the bottom of this file for an example of using the class.
+
+**Obligatory disclaimer:**
+ This is my first crack at both using xmlrpc and multi-threaded
+ programming, so there could be huge horrible bugs or design
+ flaws. If you see one, I'd love to hear about them.
+
+"""
+
+
+""" ChangeLog
+23 May 2001: Version bumped to 0.2.0
+ Numerous code and design changes
+
+21 Mar. 2001: Version bumped to 0.1.4
+ Updated rpcMixin.OnExternal to support methods with further references
+ (i.e. now you can do rpcClient.foo.bar() and have it work)
+ This probably ain't super legal in xmlrpc land, but it works just fine here
+ and we need it.
+
+6 Mar. 2001: Version bumped to 0.1.3
+ Documentation changes to make this compatible with happydoc
+
+21 Jan. 2001: Version bumped to 0.1.2
+ OnExternal() method in the mixin class now uses getattr() to check if
+ a desired method is present. It should have been done this way in
+ the first place.
+14 Dec. 2000: Version bumped to 0.1.1
+ rearranged locking code and made other changes so that multiple
+ servers in one application are possible.
+
+"""
+
+from wxPython.wx import *
+import xmlrpcserver,xmlrpclib
+import threading
+import SocketServer
+import new
+import sys
+
+rpcPENDING = 0
+rpcDONE = 1
+rpcEXCEPT = 2
+
+class RPCRequest:
+ """A wrapper to use for handling requests and their responses"""
+ status = rpcPENDING
+ result = None
+
+# here's the ID for external events
+wxEVT_EXTERNAL_EVENT = 25015
+class ExternalEvent(wxPyEvent):
+ """The custom event class used to pass xmlrpc calls from
+ the server thread into the GUI thread
+
+ """
+ def __init__(self,method,args):
+ wxPyEvent.__init__(self)
+ self.SetEventType(wxEVT_EXTERNAL_EVENT)
+ self.method = method
+ self.args = args
+ self.rpcStatus = RPCRequest()
+ self.rpcStatusLock = threading.Lock()
+ self.rpcCondVar = threading.Condition()
+
+ def Destroy(self):
+ self.method=None
+ self.args=None
+ self.rpcStatus = None
+ self.rpcStatusLock = None
+ self.rpcondVar = None
+
+def EVT_EXTERNAL_EVENT(win,func):
+ win.Connect(-1,-1,wxEVT_EXTERNAL_EVENT,func)
+
+class Handler(xmlrpcserver.RequestHandler):
+ """The handler class that the xmlrpcserver actually calls
+ when a request comes in.
+
+ """
+ def log_message(self,*args):
+ """ causes the server to stop spewing messages every time a request comes in
+
+ """
+ pass
+ def call(self,method,params):
+ """When an xmlrpc request comes in, this is the method that
+ gets called.
+
+ **Arguments**
+
+ - method: name of the method to be called
+
+ - params: arguments to that method
+
+ """
+ if method == '_rpcPing':
+ # we just acknowledge these without processing them
+ return 'ack'
+
+ # construct the event
+ evt = ExternalEvent(method,params)
+
+ # update the status variable
+ evt.rpcStatusLock.acquire()
+ evt.rpcStatus.status = rpcPENDING
+ evt.rpcStatusLock.release()
+
+ evt.rpcCondVar.acquire()
+ # dispatch the event to the GUI
+ wxPostEvent(self._app,evt)
+
+ # wait for the GUI to finish
+ while evt.rpcStatus.status == rpcPENDING:
+ evt.rpcCondVar.wait()
+ evt.rpcCondVar.release()
+ evt.rpcStatusLock.acquire()
+ if evt.rpcStatus.status == rpcEXCEPT:
+ # The GUI threw an exception, release the status lock
+ # and re-raise the exception
+ evt.rpcStatusLock.release()
+ raise evt.rpcStatus.result[0],evt.rpcStatus.result[1]
+ else:
+ # everything went through without problems
+ s = evt.rpcStatus.result
+
+ evt.rpcStatusLock.release()
+ evt.Destroy()
+ self._app = None
+ return s
+
+# this global Event is used to let the server thread
+# know when it should quit
+stopEvent = threading.Event()
+stopEvent.clear()
+
+class _ServerThread(threading.Thread):
+ """ this is the Thread class which actually runs the server
+
+ """
+ def __init__(self,server,verbose=0):
+ self._xmlServ = server
+ threading.Thread.__init__(self,verbose=verbose)
+
+ def stop(self):
+ stopEvent.set()
+
+ def shouldStop(self):
+ return stopEvent.isSet()
+
+ def run(self):
+ while not self.shouldStop():
+ self._xmlServ.handle_request()
+ self._xmlServ = None
+
+class rpcMixin:
+ """A mixin class to provide xmlrpc server functionality to wxPython
+ frames/windows
+
+ If you want to customize this, probably the best idea is to
+ override the OnExternal method, which is what's invoked when an
+ RPC is handled.
+
+ """
+
+ # we'll try a range of ports for the server, this is the size of the
+ # range to be scanned
+ nPortsToTry=20
+ if sys.platform == 'win32':
+ defPort = 800
+ else:
+ defPort = 8023
+
+ def __init__(self,host='',port=-1,verbose=0,portScan=1):
+ """Constructor
+
+ **Arguments**
+
+ - host: (optional) the hostname for the server
+
+ - port: (optional) the port the server will use
+
+ - verbose: (optional) if set, the server thread will be launched
+ in verbose mode
+
+ - portScan: (optional) if set, we'll scan across a number of ports
+ to find one which is avaiable
+
+ """
+ if port == -1:
+ port = self.defPort
+ self.verbose=verbose
+ EVT_EXTERNAL_EVENT(self,self.OnExternal)
+ if hasattr(self,'OnClose'):
+ self._origOnClose = self.OnClose
+ self.Disconnect(-1,-1,wxEVT_CLOSE_WINDOW)
+ else:
+ self._origOnClose = None
+ self.OnClose = self.RPCOnClose
+ EVT_CLOSE(self,self.RPCOnClose)
+
+ tClass = new.classobj('Handler%d'%(port),(Handler,),{})
+ tClass._app = self
+ if portScan:
+ self.rpcPort = -1
+ for i in xrange(self.nPortsToTry):
+ try:
+ xmlServ = SocketServer.TCPServer((host,port+i),tClass)
+ except:
+ pass
+ else:
+ self.rpcPort = port+i
+ else:
+ self.rpcPort = port
+ try:
+ xmlServ = SocketServer.TCPServer((host,port),tClass)
+ except:
+ self.rpcPort = -1
+
+ if self.rpcPort == -1:
+ raise 'RPCMixinError','Cannot initialize server'
+ self.servThread = _ServerThread(xmlServ,verbose=self.verbose)
+ self.servThread.setName('XML-RPC Server')
+ self.servThread.start()
+
+ def RPCOnClose(self,event):
+ """ callback for when the application is closed
+
+ be sure to shutdown the server and the server thread before
+ leaving
+
+ """
+ # by setting the global stopEvent we inform the server thread
+ # that it's time to shut down.
+ stopEvent.set()
+ if event is not None:
+ # if we came in here from a user event (as opposed to an RPC event),
+ # then we'll need to kick the server one last time in order
+ # to get that thread to terminate. do so now
+ s1 = xmlrpclib.Server('http://localhost:%d'%(self.rpcPort))
+ try:
+ s1._rpcPing()
+ except:
+ pass
+
+ if self._origOnClose is not None:
+ self._origOnClose(event)
+
+ def RPCQuit(self):
+ """ shuts down everything, including the rpc server
+
+ """
+ self.RPCOnClose(None)
+ def OnExternal(self,event):
+ """ this is the callback used to handle RPCs
+
+ **Arguments**
+
+ - event: an _ExternalEvent_ sent by the rpc server
+
+ Exceptions are caught and returned in the global _rpcStatus
+ structure. This allows the xmlrpc server to report the
+ exception to the client without mucking up any of the delicate
+ thread stuff.
+
+ """
+ event.rpcStatusLock.acquire()
+ doQuit = 0
+ try:
+ methsplit = event.method.split('.')
+ meth = self
+ for piece in methsplit:
+ meth = getattr(meth,piece)
+ except AttributeError,msg:
+ event.rpcStatus.result = 'No Such Method',msg
+ event.rpcStatus.status = rpcEXCEPT
+ else:
+ try:
+ res = apply(meth,event.args)
+ except:
+ import traceback
+ if self.verbose: traceback.print_exc()
+ event.rpcStatus.result = sys.exc_info()[:2]
+ event.rpcStatus.status = rpcEXCEPT
+ else:
+ if res is None:
+ # returning None across the xmlrpc interface is problematic
+ event.rpcStatus.result = []
+ else:
+ event.rpcStatus.result = res
+ event.rpcStatus.status = rpcDONE
+
+ event.rpcStatusLock.release()
+
+ # broadcast (using the condition var) that we're done with the event
+ event.rpcCondVar.acquire()
+ event.rpcCondVar.notify()
+ event.rpcCondVar.release()
+
+
+if __name__ == '__main__':
+ import time
+ if sys.platform == 'win32':
+ port = 800
+ else:
+ port = 8023
+
+ class rpcFrame(wxFrame,rpcMixin):
+ """A simple wxFrame with the rpcMixin functionality added
+ """
+ def __init__(self,*args,**kwargs):
+ """ rpcHost or rpcPort keyword arguments will be passed along to
+ the xmlrpc server.
+ """
+ mixinArgs = {}
+ if kwargs.has_key('rpcHost'):
+ mixinArgs['host'] = kwargs['rpcHost']
+ del kwargs['rpcHost']
+ if kwargs.has_key('rpcPort'):
+ mixinArgs['port'] = kwargs['rpcPort']
+ del kwargs['rpcPort']
+ if kwargs.has_key('rpcPortScan'):
+ mixinArgs['portScan'] = kwargs['rpcPortScan']
+ del kwargs['rpcPortScan']
+
+ apply(wxFrame.__init__,(self,)+args,kwargs)
+ apply(rpcMixin.__init__,(self,),mixinArgs)
+
+ EVT_CHAR(self,self.OnChar)
+
+ def TestFunc(self,args):
+ """a demo method"""
+ return args
+
+ def OnChar(self,event):
+ key = event.GetKeyCode()
+ if key == ord('q'):
+ self.OnQuit(event)
+
+ def OnQuit(self,event):
+ self.OnClose(event)
+
+ def OnClose(self,event):
+ self.Destroy()
+
+
+
+ class MyApp(wxApp):
+ def OnInit(self):
+ self.frame = rpcFrame(NULL, -1, "wxPython RPCDemo", wxDefaultPosition,
+ wxSize(300,300),
+ rpcHost='localhost',rpcPort=port)
+ self.frame.Show(True)
+ return True
+
+
+ def testcon(port):
+ s1 = xmlrpclib.Server('http://localhost:%d'%(port))
+ s1.SetTitle('Munged')
+ s1._rpcPing()
+ if doQuit:
+ s1.RPCQuit()
+
+ doQuit = 1
+ if len(sys.argv)>1 and sys.argv[1] == '-q':
+ doQuit = 0
+ nT = threading.activeCount()
+ app = MyApp(0)
+ activePort = app.frame.rpcPort
+ t = threading.Thread(target=lambda x=activePort:testcon(x),verbose=0)
+ t.start()
+
+ app.MainLoop()
+ # give the threads time to shut down
+ if threading.activeCount() > nT:
+ print 'waiting for all threads to terminate'
+ while threading.activeCount() > nT:
+ time.sleep(0.5)